X-Git-Url: https://hackdaworld.org/gitweb/?a=blobdiff_plain;f=betty%2Flpcload.c;h=19d5a5b2904aebc9dcdc242090ecd506331b4c95;hb=ed474ffa00e2e35babccb09d039a7a65939097a0;hp=8bd853010f225979c073ad24a3d5aaa630b693f6;hpb=e98e1b46f919aaa0eadb3a74ada0e4f6220a589c;p=my-code%2Farm.git diff --git a/betty/lpcload.c b/betty/lpcload.c index 8bd8530..19d5a5b 100644 --- a/betty/lpcload.c +++ b/betty/lpcload.c @@ -1,8 +1,10 @@ /* * lpcload.c - load firmware into ram of lpc2220 via uart0 * - * author: hackbard@hackdaworld.org + * author: hackbard@hackdaworld.org, rolf.anders@physik.uni-augsburg.de * + * build: make + * usage: sudo ./lpcload -d /dev/ttyS0 -f firmware.hex [-v] */ #include @@ -16,10 +18,24 @@ #define VERBOSE (1<<0) #define FIRMWARE (1<<1) +#define BANK0 (1<<2) +#define BANK2 (1<<3) +#define BL (1<<4) +#define BANK0_ADDR 0x80000000 +#define BANK2_ADDR 0x82000000 +#define BANK_SIZE 0x00100000 +#define BL_ADDR 0x7fffe000 +#define BL_SIZE 0x00002000 + +#define CMD_READ 'R' // stay compatible to fwflash! + +#define TXRX_TYPE_SYNC 0x00 +#define TXRX_TYPE_CKSM 0x00 #define TXRX_TYPE_BAUD 0x01 -#define TXRX_TYPE_SYNC 0x02 +#define TXRX_TYPE_CMD 0x02 #define TXRX_TYPE_DATA 0x03 +#define TXRX_TYPE_GO 0x04 #define CMD_SUCCESS "0\r\n" #define INVALID_COMMAND "1\r\n" @@ -39,10 +55,13 @@ #define INVALID_STOP_BIT "18\r\n" #define CRYSTFREQ "10000" +#define RAMOFFSET 0x40000200 -#define BUFSIZE 64 +#define BUFSIZE 128 typedef unsigned char u8; +typedef unsigned short u16; +typedef unsigned int u32; typedef struct s_lpc { int sfd; /* serial fd */ @@ -51,16 +70,24 @@ typedef struct s_lpc { char fwfile[128]; /* firmware file */ u8 info; /* info/mode */ char freq[8]; /* frequency */ - int partid; /* part id */ - u8 bcv[2]; /* boot code version */ + char bank0[127]; /* flash dump bank0 */ + int b0fd; /* dumpfile fd bank0 */ + char bank2[127]; /* flash dump bank2 */ + int b2fd; /* dumpfile fd bank0 */ + char bl[127]; /* flash dump bootloader */ + int blfd; /* dumpfile fd bootloader */ + u32 roff; /* ram offset of uc */ + u32 jaddr; /* addr for the jump */ } t_lpc; void usage(void) { printf("possible argv:\n"); - printf(" -d \n"); - printf(" -f \n"); - printf(" -c \n"); + printf(" -d \n"); + printf(" -f \n"); + printf(" -c \n"); + printf(" -Dx \n"); + printf(" x=0: bank0, x=2: bank2, x=b: bootloader\n"); printf(" -v\n"); } @@ -85,8 +112,8 @@ int open_serial_device(t_lpc *lpc) { // input/output baudrate - cfsetispeed(&term,B9600); - cfsetospeed(&term,B9600); + cfsetispeed(&term,B38400); + cfsetospeed(&term,B38400); // control options -> 8n1 @@ -99,20 +126,50 @@ int open_serial_device(t_lpc *lpc) { term.c_lflag&=~(ICANON|ECHO|ECHOE|ISIG); - // input options -> disable flow control + // input options -> enable flow control + + term.c_iflag&=~(INLCR|ICRNL|IXANY); + term.c_iflag|=(IXON|IXOFF); - term.c_iflag&=~(IXON|IXOFF|IXANY); + // output options - // more control options -> timeout + term.c_oflag=0; + + // more control options -> timeout / flow control term.c_cc[VMIN]=0; - term.c_cc[VTIME]=10; // 1 second timeout + term.c_cc[VTIME]=20; // 2 seconds timeout + //term.c_cc[VSTART]=0x11; + //term.c_cc[VSTOP]=0x13; tcsetattr(lpc->sfd,TCSANOW,&term); return lpc->sfd; } +int reconfig_serial_device(t_lpc *lpc) { + + struct termios term; + int ret; + + /* reconfigure the serial device for our lousy loader tool */ + + tcgetattr(lpc->sfd,&term); + + // disable flow control + + term.c_iflag&=~(IXON|IXOFF|IXANY|INLCR|ICRNL); + + // change baudrate + + cfsetispeed(&term,B115200); + cfsetospeed(&term,B115200); + + ret=tcsetattr(lpc->sfd,TCSANOW,&term); + + return ret; +} + int open_firmware(t_lpc *lpc) { /* open firmware file */ @@ -125,6 +182,37 @@ int open_firmware(t_lpc *lpc) { return lpc->fwfd; } +int open_dumpfiles(t_lpc *lpc) { + + /* open dumpfiles */ + + if(lpc->info&BANK0) { + lpc->b0fd=open(lpc->bank0,O_WRONLY|O_CREAT); + if(lpc->b0fd<0) { + perror("bank0 dump file open"); + return lpc->b0fd; + } + } + + if(lpc->info&BANK2) { + lpc->b2fd=open(lpc->bank2,O_WRONLY|O_CREAT); + if(lpc->b2fd<0) { + perror("bank2 dump file open"); + return lpc->b2fd; + } + } + + if(lpc->info&BL) { + lpc->blfd=open(lpc->bl,O_WRONLY|O_CREAT); + if(lpc->blfd<0) { + perror("bootloader dump file open"); + return lpc->blfd; + } + } + + return 0; + +} int txrx(t_lpc *lpc,char *buf,int len,u8 type) { int ret,cnt; @@ -149,8 +237,14 @@ int txrx(t_lpc *lpc,char *buf,int len,u8 type) { len-=ret; cnt+=ret; } - if(lpc->info&VERBOSE) - printf(" (%d)\n",cnt); + if(lpc->info&VERBOSE) { + printf(" | "); + for(i=0;iinfo&VERBOSE) - printf(" << "); + ret=read(lpc->sfd,buf,1); + if(ret<0) { + perror("txrx read (first byte)"); + return ret; + } + + switch(buf[0]) { + case 'S': + cnt=13; + break; + case 'O': + cnt=3; + break; + case 'R': + cnt=7; + break; + case '0': + cnt=2; + break; + default: + printf("txrx read: bad return byte '%02x'\n",buf[0]); + break; + } + ret=1; - cnt=0; - while(ret>0) { - ret=read(lpc->sfd,buf+cnt,BUFSIZE-cnt); + i=cnt; + while(i) { + ret=read(lpc->sfd,buf+1+cnt-i,i); if(ret<0) { - perror("txrx read"); + perror("txrx read (next bytes)"); return ret; } - if(ret+cnt>BUFSIZE) { - printf("txrx read: too small buf size (%d)!\n",BUFSIZE); - return -1; - } - if(lpc->info&VERBOSE) - for(i=0;i0x19)&(buf[cnt+i]<0x7f))? - buf[cnt+i]:'.'); - cnt+=ret; + i-=ret; } - if(lpc->info&VERBOSE) - printf(" (%d)\n",cnt); - len=cnt; - buf[cnt]='\0'; + if(lpc->info&VERBOSE) { + printf(" << "); + for(i=0;i0x19)&(buf[i]<0x7f))? + buf[i]:'.'); + printf(" | "); + for(i=0;ipartid=atoi(buf); + memcpy(buf,"U 23130\r\n",9); + ret=txrx(lpc,buf,9,TXRX_TYPE_CMD); - return lpc->partid; + return ret; } -int read_bcv(t_lpc *lpc) { +int go(t_lpc *lpc) { char buf[BUFSIZE]; - char *ptr; + int ret,len; - memcpy(buf,"K\r\n",3); - txrx(lpc,buf,3,TXRX_TYPE_DATA); - ptr=strtok(buf,"\r\n"); - lpc->bcv[0]=strtol(ptr,NULL,16); - ptr=strtok(NULL,"\r\n"); - lpc->bcv[1]=strtol(ptr,NULL,16); + snprintf(buf,BUFSIZE,"G %d A\r\n",lpc->jaddr); + len=strlen(buf); + ret=txrx(lpc,buf,len,TXRX_TYPE_GO); - return 0; + return ret; } -int uuencode(char *in,char *out) { +int uuencode(u8 *in,u8 *out,int len) { - out[0]=0x20+((in[0]>>2)&0x3f); - out[1]=0x20+(((in[0]<<4)|(in[1]>>4))&0x3f); - out[2]=0x20+(((in[1]<<2)|(in[2]>>6))&0x3f); - out[3]=0x20+(in[2]&0x3f); + out[0]=0x20+len; + out[1]=0x20+((in[0]>>2)&0x3f); + out[2]=0x20+(((in[0]<<4)|(in[1]>>4))&0x3f); + out[3]=0x20+(((in[1]<<2)|(in[2]>>6))&0x3f); + out[4]=0x20+(in[2]&0x3f); return 0; } -int write_to_ram(t_lpc *lpc,char *buf,int addr,int len) { +int write_to_ram(t_lpc *lpc,char *buf,u32 addr,int len) { int lcount; - u8 checksum; + u32 checksum; + char txrxbuf[BUFSIZE]; + int count,bcnt; + int nlen,slen; + int i; + /* check length */ if(len%4) { printf("ram write: not a multiple of 4\n"); return -1; } - while(1) { - if(!(lcount%20)) { + /* make it a multiple of 3 (reason: uuencode) */ + nlen=(!(len%3))?len:((len/3+1)*3); + if(nlen>BUFSIZE) { + printf("ram write: too much data\n"); + return -1; + } + for(i=len;iroff; + + /* prepare write command */ + if(lpc->info&VERBOSE) + printf("writing 0x%02x bytes to 0x%08x\n",len,addr); + snprintf(txrxbuf,BUFSIZE,"W %d %d\r\n",addr,len); + slen=strlen(txrxbuf); + + /* send command and check return code */ + txrx(lpc,txrxbuf,slen,TXRX_TYPE_CMD); + + /* send data */ + lcount=0; + bcnt=0; + count=0; + checksum=0; + while(bcntsfd,txrxbuf+4,4); + printf("ram write: resending ...\n"); + bcnt-=count; } - /* reset checksum */ + if(strncmp(txrxbuf,"OK\r\n",4)) { + printf("ram write: bad response\n"); + return -1; + } + /* reset checksum & counter */ checksum=0; + count=0; } - lcount+=1; } return 0; @@ -309,7 +492,152 @@ int write_to_ram(t_lpc *lpc,char *buf,int addr,int len) { int firmware_to_ram(t_lpc *lpc) { + char buf[BUFSIZE]; + u32 addr,len,type; + int ret,temp; + /* read a line */ + ret=1; + while(ret) { + /* sync line */ + ret=read(lpc->fwfd,buf,1); + switch(buf[0]) { + case '\r': + continue; + case '\n': + continue; + case ':': + /* start code */ + break; + default: + printf("fw to ram: no ihex format\n"); + return -1; + } + /* read len */ + ret=read(lpc->fwfd,buf,2); + sscanf(buf,"%02x",&len); + /* read addr */ + ret=read(lpc->fwfd,buf,4); + sscanf(buf,"%04x",&addr); + /* read type */ + ret=read(lpc->fwfd,buf,2); + sscanf(buf,"%02x",&type); + /* successfull return if type is end of file */ + if(type==0x01) + return 0; + /* read data (and cksum) */ + ret=read(lpc->fwfd,buf,2*(len+1)); + if(ret!=(2*(len+1))) { + printf("fw to ram: data missing\n"); + return -1; + } + for(ret=0;retroff=((buf[0]<<24)|(buf[1]<<16)); + break; + case 0x05: + lpc->jaddr=((buf[0]<<24)|(buf[1]<<16)); + lpc->jaddr|=((buf[2]<<8)|buf[3]); + break; + default: + printf("fw to ram: unknown type %02x\n",type); + return -1; + } + } + + return 0; +} + +int lpc_txbuf_flush(t_lpc *lpc) { + + int i,ret; + u8 buf[16]; + + ret=1; + printf("flushing lpc tx buffer: "); + while(ret) { + ret=read(lpc->sfd,buf,16); + for(i=0;i>24)&0xff; + buf[2]=(addr>>16)&0xff; + buf[3]=(addr>>8)&0xff; + buf[4]=addr&0xff; + buf[5]=(len>>24)&0xff; + buf[6]=(len>>16)&0xff; + buf[7]=(len>>8)&0xff; + buf[8]=len&0xff; + printf(" sending cmd: "); + while(size) { + ret=write(sfd,buf+cnt,size); + for(i=cnt;i aborting\n"); + goto end; + } - // to be continued ... (parsing fw file and poking it to ram) + /* open firmware file */ if(open_firmware(&lpc)<0) goto end; + + /* open dump files */ + if(open_dumpfiles(&lpc)<0) + goto end; + + /* parse intel hex file and write to ram */ + printf("write firmware to ram ...\n"); firmware_to_ram(&lpc); + /* unlock go cmd */ + printf("unlock go command ...\n"); + unlock_go(&lpc); + + /* go! */ + printf("go ...\n"); + ret=go(&lpc); + + /* flush the lpc2220 tx buf */ + lpc_txbuf_flush(&lpc); + + /* reconfigure the serial port */ + if(reconfig_serial_device(&lpc)<0) + goto end; + + /* download flash/bootloader content */ + if(lpc.info&BANK0) + dump_files(lpc.sfd,lpc.b0fd,BANK0_ADDR,BANK_SIZE); + if(lpc.info&BANK2) + dump_files(lpc.sfd,lpc.b2fd,BANK2_ADDR,BANK_SIZE); + if(lpc.info&BL) + dump_files(lpc.sfd,lpc.blfd,BL_ADDR,BL_SIZE); + end: - close(lpc.sfd); - close(lpc.fwfd); + if(lpc.sfd) + close(lpc.sfd); + if(lpc.fwfd) + close(lpc.fwfd); + if(lpc.b0fd) + close(lpc.b0fd); + if(lpc.b2fd) + close(lpc.b2fd); + if(lpc.blfd) + close(lpc.blfd); return 0; }