summaryrefslogtreecommitdiff
path: root/digital/avr/modules/usb/lufa/Projects/AVRISP_Programmer/AVRISP_Firmware_Design.txt
blob: 643454d4815bc1ce94b231cfc77506fedcfce627 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
Instructions for converting the LUFA USBtoSerial Demo to an AVR ISP Programmer.
By Opendous Inc., Copyright under the Creative Commons Attribution License:
http://creativecommons.org/licenses/by/3.0/

1) Start with the LUFA/Demos/USBtoSerial firmware.
    - rename USBtoSerial.c, USBtoSerial.h, and USBtoSerial.aps to
      AVRISP_Programmer.*
    - edit AVRISP_Programmer.aps and rename all instances of "USBtoSerial" to
      "AVRISP_Programmer"
    - copy AVRISP_Programmer.txt from an older version of AVRISP_Programmer

2) Edit makefile by changing TARGET from "USBtoSerial" to "AVRISP_Programmer"

3) Edit AVRISP_Programmer.h:
    - change ifdef _USB_TO_SERIAL_H to _AVRISP_PROGRAMMER_H_
    - rename ReconfigureUSART(void) to ReconfigureSPI(void)
    - add  void processHostSPIRequest(void);  &  void delay_ms(uint8_t dly);
    - replace the define for Serial.h with one for SPI.h:
      #include <libs/LUFA/Drivers/AT90USBXXX/SPI.h>

4) Make alterations to Descriptors.c
    - change manufacturer string to "www.AVRopendous.org", length=19
    - change product string to "LUFA-Based AVR ISP Programmer", length=29

5) Edit Ringbuff.h to enable the Peek Command:  #define BUFF_USEPEEK

6) Edit AVRISP_Programmer.c:
    - change #include "USBtoSerial.h"  to  #include "AVRISP_Programmer.h"
    - change BUTTLOADTAG(ProjName to "LUFA AVR910 ISP Programmer"
    - in main(), rename ReconfigureUSART() to Reconfigure();
    - in EVENT_HANDLER(USB_UnhandledControlPacket), rename ReconfigureUSART
    - delete the ISRs: ISR(USART1_RX_vect)  &  ISR(USART1_TX_vect)
    - delete ReconfigureUSART(void)
    - add  void ReconfigureSPI(void),  void processHostSPIRequest(void),
      and  void delay_ms(uint8_t dly)  from a previous version
    - add Timer1 and SPI initialization code to main():
	    /* Hardware Initialization */
	    //LEDs_Init();
	    DDRB = 0;
	    PORTB = 0;
	    DDRC |= ((1 << PC2) | (1 << PC4) | (1 << PC5) | (1 << PC6) | (1 << PC7)); //AT90USBxx2
	    // PC2 is also used for RESET, so set it HIGH initially - note 'P' command sets it to LOW (Active)
	    PORTC |= ((1 << PC2) | (1 << PC4) | (1 << PC5) | (1 << PC6) | (1 << PC7)); //AT90USBxx2
	    DDRD = 0;
	    PORTD = (1 << PB7); // only PB7(HWB) should be High as this is the bootloader pin
	    // Prepare PortB for SPI - set PB0(^SS), PB1(SCK), PB2(MOSI) as output as well as all other pins except PB3(MISO)
	    DDRB = (1 << PB0) | (1 << PB1) | (1 << PB2) | (0 << PB3) | (1 << PB4) | (1 << PB5) | (1 << PB6) | (1 << PB7);
	    PORTB |= (1 << PB0);

	    // initialize Timer1 for use in delay function
	    TCCR1A = 0;
	    //TCCR1B = (1 << CS10); // no prescaling, use CLK
	    TCCR1B = ((1 << CS12) | (1 << CS10)); // prescale by CLK/1024
	    // 8MHz/1024 = 7813 ticks per second --> ~8 ticks per millisecond (ms)
	    timerval = TCNT1; // start timer1

    - In TASK(CDC_Task) in the
        if (USB_IsConnected) {
            if (Endpoint_ReadWriteAllowed()) {
                while (Endpoint_BytesInEndpoint()) {
                    ...
      structure, after  Buffer_StoreElement(&Rx_Buffer, Endpoint_Read_Byte()):

        /* Each time there is an element, check which comand should be
	        run and if enough data is available to run that command.
	        There are 1-byte, 2-byte, 3-byte, 4-byte commands, and 5-byte commands
	        Remember that the "which command" byte counts as 1 */
        if (Rx_Buffer.Elements == 0) {
	        // do nothing, wait for data
        } else {
	        tempByte = Buffer_PeekElement(&Rx_Buffer); // peek at first element

		        /* make sure the issued command and associated data are all ready */
	        if (Rx_Buffer.Elements == 1) { // zero data byte command
		        if ((tempByte == 'P') | (tempByte == 'a') | (tempByte == 'm') |
		        (tempByte == 'R') | (tempByte == 'd') | (tempByte == 'e') |
		        (tempByte == 'L') | (tempByte == 's') | (tempByte == 't') | 
		        (tempByte == 'S') | (tempByte == 'V') | (tempByte == 'v') |
		        (tempByte == 'p') | (tempByte == 'F')) {
            	processHostSPIRequest(); // command has enough data, process it
		        }
	        } else if (Rx_Buffer.Elements == 2) { // one data byte command
		        if ((tempByte == 'T') | (tempByte == 'c') | (tempByte == 'C') |
			        (tempByte == 'D') | (tempByte == 'l') | (tempByte == 'f') |
			        (tempByte == 'x') | (tempByte == 'y')) {
			        processHostSPIRequest(); // command has enough data, process it
		        }
	        } else if (Rx_Buffer.Elements == 3) { // two data byte command
		        if ((tempByte == 'A') | (tempByte == 'Z')) {
			        processHostSPIRequest(); // command has enough data, process it
		        }
	        } else if (Rx_Buffer.Elements == 4) { // three data byte command
		        if ((tempByte == ':')) {
            	processHostSPIRequest(); // command has enough data, process it
		        }
	        } else if (Rx_Buffer.Elements == 5) { // four data byte command
		        if ((tempByte == '.')) {
			        processHostSPIRequest(); // command has enough data, process it
		        }
	        } else {
		        // do nothing
	        }
        }

    - need to add code to flush the buffer.  Change:
        /* Check if Rx buffer contains data */
		if (Rx_Buffer.Elements)
		{
			/* Initiate the transmission of the buffer contents if USART idle*/
			if (!(Transmitting))
			{
				Transmitting = true;
				Serial_TxByte(Buffer_GetElement(&Rx_Buffer));
			}
		}
      To:
        /* Check if Rx buffer contains data */
		if (Rx_Buffer.Elements)
		{
			/* Initiate the transmission of the buffer contents if USART idle*/
			if (!(Transmitting))
			{
				Transmitting = true;
				/* The following flushes the receive buffer to prepare for new
                   data and commands. Need to flush the buffer as the command
                   byte which is peeked above needs to be dealt with, otherwise
                   the command bytes will overflow the buffer eventually */
				//Buffer_GetElement(&Rx_Buffer); // also works
				Buffer_Initialize(&Rx_Buffer);
			}
		}

    - need to add the following defines and globals:
        #define RESETPORT       PORTB
        #define RESETPIN        PB0
        #define RESETPORT2      PORTC
        #define RESETPIN2       PC2
        #define CR_HEX          '\r'

        #define DELAY_VERYSHORT    0x01
        #define DELAY_SHORT        0x02
        #define DELAY_MEDIUM       0x03
        #define DELAY_LONG         0x05
        #define DELAY_MULTIPLE     0x04

        /*  AVR Device Codes - Can have a maximum of 14 but can be any you want.
            Note that these are completely irrelevent.  If AVRdude supports a
            device, then that device is programmable.  Use -F switch to ignore
            device codes. */
        #define AVRDEVCODE01	0x55 /* ATtiny12 */
        #define AVRDEVCODE02	0x56 /* ATtiny15 */
        #define AVRDEVCODE03	0x5E /* ATtiny261 */
        #define AVRDEVCODE04	0x76 /* ATmega8 */
        #define AVRDEVCODE05	0x74 /* ATmega16 */
        #define AVRDEVCODE06	0x72 /* ATmega32 */
        #define AVRDEVCODE07	0x45 /* ATmega64 */
        #define AVRDEVCODE08	0x74 /* ATmega644 */
        #define AVRDEVCODE09	0x43 /* ATmega128 */
        #define AVRDEVCODE10	0x63 /* ATmega162 */
        #define AVRDEVCODE11	0x78 /* ATmega169 */
        #define AVRDEVCODE12	0x6C /* AT90S4434 */
        #define AVRDEVCODE13	0x38 /* AT90S8515A */
        #define AVRDEVCODE14	0x65 /* AT90S8555 */

        /* some global variables used throughout */
        uint8_t tempIOreg = 0;
        uint8_t tempIOreg2 = 0;
        uint8_t tempIOreg3 = 0;
        uint8_t tempIOreg4 = 0;
        uint8_t dataWidth = 0;
        uint8_t firstRun = 1;
        uint8_t deviceCode = 0;
        uint8_t tempByte = 0;
        uint16_t currAddress = 0;
        uint16_t timerval = 0;