First CVS version
This commit is contained in:
299
bootloader/main.c
Normal file
299
bootloader/main.c
Normal file
@ -0,0 +1,299 @@
|
||||
/* Name: main.c
|
||||
* Project: USBaspLoader
|
||||
* Author: Christian Starkjohann
|
||||
* Creation Date: 2007-12-08
|
||||
* Tabsize: 4
|
||||
* Copyright: (c) 2007 by OBJECTIVE DEVELOPMENT Software GmbH
|
||||
* License: GNU GPL v2 (see License.txt)
|
||||
* This Revision: $Id: main.c,v 1.1 2008/07/09 20:47:11 rschaten Exp $
|
||||
*/
|
||||
|
||||
#include <avr/io.h>
|
||||
#include <avr/interrupt.h>
|
||||
#include <avr/pgmspace.h>
|
||||
#include <avr/wdt.h>
|
||||
#include <avr/boot.h>
|
||||
#include <avr/eeprom.h>
|
||||
#include <util/delay.h>
|
||||
#include <string.h>
|
||||
|
||||
static void leaveBootloader() __attribute__((__noreturn__));
|
||||
|
||||
#include "bootloaderconfig.h"
|
||||
#include "usbdrv/usbdrv.c"
|
||||
|
||||
/* ------------------------------------------------------------------------ */
|
||||
|
||||
/* Request constants used by USBasp */
|
||||
#define USBASP_FUNC_CONNECT 1
|
||||
#define USBASP_FUNC_DISCONNECT 2
|
||||
#define USBASP_FUNC_TRANSMIT 3
|
||||
#define USBASP_FUNC_READFLASH 4
|
||||
#define USBASP_FUNC_ENABLEPROG 5
|
||||
#define USBASP_FUNC_WRITEFLASH 6
|
||||
#define USBASP_FUNC_READEEPROM 7
|
||||
#define USBASP_FUNC_WRITEEEPROM 8
|
||||
#define USBASP_FUNC_SETLONGADDRESS 9
|
||||
|
||||
/* ------------------------------------------------------------------------ */
|
||||
|
||||
#ifndef ulong
|
||||
# define ulong unsigned long
|
||||
#endif
|
||||
#ifndef uint
|
||||
# define uint unsigned int
|
||||
#endif
|
||||
|
||||
/* defaults if not in config file: */
|
||||
#ifndef HAVE_EEPROM_PAGED_ACCESS
|
||||
# define HAVE_EEPROM_PAGED_ACCESS 0
|
||||
#endif
|
||||
#ifndef HAVE_EEPROM_BYTE_ACCESS
|
||||
# define HAVE_EEPROM_BYTE_ACCESS 0
|
||||
#endif
|
||||
#ifndef BOOTLOADER_CAN_EXIT
|
||||
# define BOOTLOADER_CAN_EXIT 0
|
||||
#endif
|
||||
|
||||
/* allow compatibility with avrusbboot's bootloaderconfig.h: */
|
||||
#ifdef BOOTLOADER_INIT
|
||||
# define bootLoaderInit() BOOTLOADER_INIT
|
||||
# define bootLoaderExit()
|
||||
#endif
|
||||
#ifdef BOOTLOADER_CONDITION
|
||||
# define bootLoaderCondition() BOOTLOADER_CONDITION
|
||||
#endif
|
||||
|
||||
/* device compatibility: */
|
||||
#ifndef GICR /* ATMega*8 don't have GICR, use MCUCR instead */
|
||||
# define GICR MCUCR
|
||||
#endif
|
||||
|
||||
/* ------------------------------------------------------------------------ */
|
||||
|
||||
typedef union longConverter{
|
||||
ulong l;
|
||||
uint w[2];
|
||||
uchar b[4];
|
||||
}longConverter_t;
|
||||
|
||||
static uchar requestBootLoaderExit;
|
||||
static longConverter_t currentAddress; /* in bytes */
|
||||
static uchar bytesRemaining;
|
||||
static uchar isLastPage;
|
||||
#if HAVE_EEPROM_PAGED_ACCESS
|
||||
static uchar currentRequest;
|
||||
#else
|
||||
static const uchar currentRequest = 0;
|
||||
#endif
|
||||
|
||||
static const uchar signatureBytes[4] = {
|
||||
#ifdef SIGNATURE_BYTES
|
||||
SIGNATURE_BYTES
|
||||
#elif defined (__AVR_ATmega8__) || defined (__AVR_ATmega8HVA__)
|
||||
0x1e, 0x93, 0x07, 0
|
||||
#elif defined (__AVR_ATmega48__) || defined (__AVR_ATmega48P__)
|
||||
0x1e, 0x92, 0x05, 0
|
||||
#elif defined (__AVR_ATmega88__) || defined (__AVR_ATmega88P__)
|
||||
0x1e, 0x93, 0x0a, 0
|
||||
#elif defined (__AVR_ATmega168__) || defined (__AVR_ATmega168P__)
|
||||
0x1e, 0x94, 0x06, 0
|
||||
#elif defined (__AVR_ATmega328P__)
|
||||
0x1e, 0x95, 0x0f, 0
|
||||
#elif defined (__AVR_ATmega32__)
|
||||
0x1e, 0x95, 0x02, 0
|
||||
#else
|
||||
# error "Device signature is not known, please edit main.c!"
|
||||
#endif
|
||||
};
|
||||
|
||||
#if (FLASHEND) > 0xffff /* we need long addressing */
|
||||
# define CURRENT_ADDRESS currentAddress.l
|
||||
#else
|
||||
# define CURRENT_ADDRESS currentAddress.w[0]
|
||||
#endif
|
||||
|
||||
/* ------------------------------------------------------------------------ */
|
||||
|
||||
static void (*nullVector)(void) __attribute__((__noreturn__));
|
||||
|
||||
static void leaveBootloader()
|
||||
{
|
||||
DBG1(0x01, 0, 0);
|
||||
bootLoaderExit();
|
||||
cli();
|
||||
GICR = (1 << IVCE); /* enable change of interrupt vectors */
|
||||
GICR = (0 << IVSEL); /* move interrupts to application flash section */
|
||||
/* We must go through a global function pointer variable instead of writing
|
||||
* ((void (*)(void))0)();
|
||||
* because the compiler optimizes a constant 0 to "rcall 0" which is not
|
||||
* handled correctly by the assembler.
|
||||
*/
|
||||
nullVector();
|
||||
}
|
||||
|
||||
/* ------------------------------------------------------------------------ */
|
||||
|
||||
uchar usbFunctionSetup(uchar data[8])
|
||||
{
|
||||
usbRequest_t *rq = (void *)data;
|
||||
uchar len = 0;
|
||||
static uchar replyBuffer[4];
|
||||
|
||||
usbMsgPtr = replyBuffer;
|
||||
if(rq->bRequest == USBASP_FUNC_TRANSMIT){ /* emulate parts of ISP protocol */
|
||||
uchar rval = 0;
|
||||
usbWord_t address;
|
||||
address.bytes[1] = rq->wValue.bytes[1];
|
||||
address.bytes[0] = rq->wIndex.bytes[0];
|
||||
if(rq->wValue.bytes[0] == 0x30){ /* read signature */
|
||||
rval = rq->wIndex.bytes[0] & 3;
|
||||
rval = signatureBytes[rval];
|
||||
#if HAVE_EEPROM_BYTE_ACCESS
|
||||
}else if(rq->wValue.bytes[0] == 0xa0){ /* read EEPROM byte */
|
||||
rval = eeprom_read_byte((void *)address.word);
|
||||
}else if(rq->wValue.bytes[0] == 0xc0){ /* write EEPROM byte */
|
||||
eeprom_write_byte((void *)address.word, rq->wIndex.bytes[1]);
|
||||
#endif
|
||||
}else{
|
||||
/* ignore all others, return default value == 0 */
|
||||
}
|
||||
replyBuffer[3] = rval;
|
||||
len = 4;
|
||||
}else if(rq->bRequest == USBASP_FUNC_ENABLEPROG){
|
||||
/* replyBuffer[0] = 0; is never touched and thus always 0 which means success */
|
||||
len = 1;
|
||||
}else if(rq->bRequest >= USBASP_FUNC_READFLASH && rq->bRequest <= USBASP_FUNC_SETLONGADDRESS){
|
||||
currentAddress.w[0] = rq->wValue.word;
|
||||
if(rq->bRequest == USBASP_FUNC_SETLONGADDRESS){
|
||||
#if (FLASHEND) > 0xffff
|
||||
currentAddress.w[1] = rq->wIndex.word;
|
||||
#endif
|
||||
}else{
|
||||
bytesRemaining = rq->wLength.bytes[0];
|
||||
/* if(rq->bRequest == USBASP_FUNC_WRITEFLASH) only evaluated during writeFlash anyway */
|
||||
isLastPage = rq->wIndex.bytes[1] & 0x02;
|
||||
#if HAVE_EEPROM_PAGED_ACCESS
|
||||
currentRequest = rq->bRequest;
|
||||
#endif
|
||||
len = 0xff; /* hand over to usbFunctionRead() / usbFunctionWrite() */
|
||||
}
|
||||
#if BOOTLOADER_CAN_EXIT
|
||||
}else if(rq->bRequest == USBASP_FUNC_DISCONNECT){
|
||||
requestBootLoaderExit = 1; /* allow proper shutdown/close of connection */
|
||||
#endif
|
||||
}else{
|
||||
/* ignore: USBASP_FUNC_CONNECT */
|
||||
}
|
||||
return len;
|
||||
}
|
||||
|
||||
uchar usbFunctionWrite(uchar *data, uchar len)
|
||||
{
|
||||
uchar isLastWrite;
|
||||
|
||||
DBG1(0x31, (void *)¤tAddress.l, 4);
|
||||
if(len > bytesRemaining)
|
||||
len = bytesRemaining;
|
||||
bytesRemaining -= len;
|
||||
isLastWrite = bytesRemaining == 0;
|
||||
if(currentRequest >= USBASP_FUNC_READEEPROM){
|
||||
eeprom_write_block(data, (void *)currentAddress.w[0], len);
|
||||
currentAddress.w[0] += len;
|
||||
}else{
|
||||
char i = len;
|
||||
while(i > 0){
|
||||
i -= 2;
|
||||
if((currentAddress.w[0] & (SPM_PAGESIZE - 1)) == 0){ /* if page start: erase */
|
||||
DBG1(0x33, 0, 0);
|
||||
#ifndef NO_FLASH_WRITE
|
||||
cli();
|
||||
boot_page_erase(CURRENT_ADDRESS); /* erase page */
|
||||
sei();
|
||||
boot_spm_busy_wait(); /* wait until page is erased */
|
||||
#endif
|
||||
}
|
||||
DBG1(0x32, 0, 0);
|
||||
cli();
|
||||
boot_page_fill(CURRENT_ADDRESS, *(short *)data);
|
||||
sei();
|
||||
CURRENT_ADDRESS += 2;
|
||||
data += 2;
|
||||
/* write page when we cross page boundary or we have the last partial page */
|
||||
if((currentAddress.w[0] & (SPM_PAGESIZE - 1)) == 0 || (i <= 0 && isLastWrite && isLastPage)){
|
||||
DBG1(0x34, 0, 0);
|
||||
#ifndef NO_FLASH_WRITE
|
||||
cli();
|
||||
boot_page_write(CURRENT_ADDRESS - 2);
|
||||
sei();
|
||||
boot_spm_busy_wait();
|
||||
cli();
|
||||
boot_rww_enable();
|
||||
sei();
|
||||
#endif
|
||||
}
|
||||
}
|
||||
DBG1(0x35, (void *)¤tAddress.l, 4);
|
||||
}
|
||||
return isLastWrite;
|
||||
}
|
||||
|
||||
uchar usbFunctionRead(uchar *data, uchar len)
|
||||
{
|
||||
if(len > bytesRemaining)
|
||||
len = bytesRemaining;
|
||||
bytesRemaining -= len;
|
||||
if(currentRequest >= USBASP_FUNC_READEEPROM){
|
||||
eeprom_read_block(data, (void *)currentAddress.w[0], len);
|
||||
}else{
|
||||
memcpy_P(data, (PGM_VOID_P)CURRENT_ADDRESS, len);
|
||||
}
|
||||
CURRENT_ADDRESS += len;
|
||||
return len;
|
||||
}
|
||||
|
||||
/* ------------------------------------------------------------------------ */
|
||||
|
||||
static void initForUsbConnectivity(void)
|
||||
{
|
||||
uchar i = 0;
|
||||
|
||||
usbInit();
|
||||
/* enforce USB re-enumerate: */
|
||||
usbDeviceDisconnect(); /* do this while interrupts are disabled */
|
||||
while(--i){ /* fake USB disconnect for > 250 ms */
|
||||
wdt_reset();
|
||||
_delay_ms(1);
|
||||
}
|
||||
usbDeviceConnect();
|
||||
sei();
|
||||
}
|
||||
|
||||
int main(void)
|
||||
{
|
||||
/* initialize */
|
||||
bootLoaderInit();
|
||||
odDebugInit();
|
||||
DBG1(0x00, 0, 0);
|
||||
#ifndef NO_FLASH_WRITE
|
||||
GICR = (1 << IVCE); /* enable change of interrupt vectors */
|
||||
GICR = (1 << IVSEL); /* move interrupts to boot flash section */
|
||||
#endif
|
||||
if(bootLoaderCondition()){
|
||||
uint i = 0;
|
||||
initForUsbConnectivity();
|
||||
do{
|
||||
usbPoll();
|
||||
#if BOOTLOADER_CAN_EXIT
|
||||
if(requestBootLoaderExit){
|
||||
if(--i == 0)
|
||||
break;
|
||||
}
|
||||
#endif
|
||||
}while(bootLoaderCondition()); /* main event loop */
|
||||
}
|
||||
leaveBootloader();
|
||||
return 0;
|
||||
}
|
||||
|
||||
/* ------------------------------------------------------------------------ */
|
Reference in New Issue
Block a user