⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 avrdummy.c

📁 用于传感器网络的节点操作系统 TinyOS 结构设计非常有意思
💻 C
字号:
/*	AvrDummy.C		Dummy device driver for the AVR parallel access	Uros Platise (c) 1999*/#ifndef NO_DAPA#include "timeradd.h"#include "AvrDummy.h"/* Private Functions*/void TAvrDummy::EnableAvr(){  unsigned char prg  [4] = { 0xAC, 0x53, 0, 0 };  int try_number = 32;  bool no_retry = GetCmdParam("-dno-retry", false);  const char *part_name = GetCmdParam("-dpart");  if (part_name && strcasecmp(part_name, "at90s1200") == 0)    no_retry = true;  /* XXX */  /* Enable AVR programming mode */  do{    prg[0]=0xAC; prg[1]=0x53; prg[2]=prg[3]=0;    Send(prg, 4);    if (no_retry) break;    if (prg[2] == 0x53) break;    PulseSck();  } while (try_number--);    if (try_number>=0){    Info(2,"AVR Direct Parallel Access succeeded after %d retries.\n",       32-try_number);  } else {    Info(2,"AVR Direct Parallel Access failed after 32 retries.\n");  }    /* Get AVR Info */  vendor_code = GetPartInfo(0);  part_family = GetPartInfo(1);  part_number = GetPartInfo(2);  if (part_name)    OverridePart(part_name);  Identify();}TByteTAvrDummy::GetPartInfo(TAddr addr){  TByte info [4] = { 0x30, 0, addr, 0 };  Send(info, 4);  return info[3];}voidTAvrDummy::WriteProgramMemoryPage(){  struct timeval t_start_wr, t_start_poll, t_wait, t_timeout, t_end, t_write;  bool poll_data = use_data_polling && TestFeatures(AVR_PAGE_POLL)		   && (page_poll_byte != 0xFF);  TByte prg_page [4] = { 0x4C,			(TByte)((page_addr >> 9) & 0xff),			(TByte)((page_addr >> 1) & 0xff),			0 };  gettimeofday(&t_start_wr, NULL);  t_wait.tv_sec = 0;  t_wait.tv_usec = Get_t_wd_flash();  Info(4, "Programming page address: %d (%.2x, %.2x, %.2x, %.2x)\n",     page_addr, prg_page[0], prg_page[1], prg_page[2], prg_page[3]);  Send(prg_page, 4);  gettimeofday(&t_start_poll, NULL);  timeradd(&t_start_poll, &t_wait, &t_timeout);  /* Wait */  do {    gettimeofday(&t_end, NULL);    if (poll_data) {      TByte rbyte = ReadByte(page_poll_addr);      if (rbyte == page_poll_byte)	break;    }  } while (timercmp(&t_end, &t_timeout, <));  /* Write Statistics */  timersub(&t_end, &t_start_wr, &t_write);  /* t_write = t_end - t_start_wr */  if (poll_data) {    float write_time = 1.0e-6 * t_write.tv_usec + t_write.tv_sec;    total_poll_time += write_time;    if (max_poll_time < write_time)      max_poll_time = write_time;    if (min_poll_time > write_time)      min_poll_time = write_time;    total_poll_cnt++;  }  page_addr_fetched=false;  page_poll_byte = 0xFF;}/* Device Interface Functions*/TByteTAvrDummy::ReadByte(TAddr addr){  TByte readback = 0xFF;      CheckMemoryRange(addr);  if (segment == SEG_FLASH) {    TByte hl = (addr & 1) ? 0x28 : 0x20;    TByte flash[4] = { hl,		       (TByte)((addr >> 9) & 0xff),		       (TByte)((addr >> 1) & 0xff),			 0 };    Send(flash, 4);    readback = flash[3];  } else if (segment == SEG_EEPROM) {    TByte eeprom [4] = { 0xA0, 			 (TByte)((addr>>8)&0xff), 			 (TByte)(addr&0xff), 			 0 };    Send(eeprom, 4);    readback = eeprom[3];  } else if (segment==SEG_FUSE) {    switch (addr) {    case 0:      if (TestFeatures(AVR_FUSE_RD))	readback = ReadFuseLowBits();      else if (TestFeatures(AVR_LOCK_RD76))	readback = ReadLockFuseBits();      break;    case 1:      if (TestFeatures(AVR_FUSE_HIGH))	readback = ReadFuseHighBits();      break;    case 2:      if (TestFeatures(AVR_CAL_RD))	readback = ReadCalByte();      break;    case 3:      readback = ReadLockBits();      break;    }    Info(3, "Read fuse/cal/lock: byte %d = 0x%02X\n",	 (int) addr, (int) readback);  }  return readback;}/* Read Lock/Fuse Bits:           7     6     5     4     3     2     1     0 2333,4433,m103,m603,tn12,tn15: x     x     x     x     x     LB2   LB1   x 2323,8535:                     LB1   LB2   SPIEN x     x     x     x     FSTRT 2343:                          LB1   LB2   SPIEN x     x     x     x     RCEN tn22:                          LB1   LB2   SPIEN x     x     x     x     0 m161,m163,m323,m128:           x     x     BLB12 BLB11 BLB02 BLB01 LB2   LB1 */TByteTAvrDummy::ReadLockFuseBits(){  TByte lockfuse[4] = { 0x58, 0, 0, 0 };  Send(lockfuse, 4);  return lockfuse[3];}/* Read Fuse Bits (Low):          7     6     5     4     3     2     1     0 2333,4433:                     x     x     SPIEN BODLV BODEN CKSL2 CKSL1 CKSL0 m103,m603:                     x     x     SPIEN x     EESAV 1     SUT1  SUT0 tn12:                          BODLV BODEN SPIEN RSTDI CKSL3 CKSL2 CKSL1 CKSL0 tn15:                          BODLV BODEN SPIEN RSTDI x     x     CKSL1 CKSL0 m161:                          x     BTRST SPIEN BODLV BODEN CKSL2 CKSL1 CKSL0 m163,m323:                     BODLV BODEN x     x     CKSL3 CKSL2 CKSL1 CKSL0 m128:                          BODLV BODEN SUT1  SUT0  CKSL3 CKSL2 CKSL1 CKSL0 */TByteTAvrDummy::ReadFuseLowBits(){  TByte fuselow[4] = { 0x50, 0, 0, 0 };  Send(fuselow, 4);  return fuselow[3];}/* Read Fuse Bits High:           7     6     5     4     3     2     1     0 m163:                          x     x     x     x     1     BTSZ1 BTSZ0 BTRST m323:                          OCDEN JTGEN x     x     EESAV BTSZ1 BTSZ0 BTRST m128:                          OCDEN JTGEN x     CKOPT EESAV BTSZ1 BTSZ0 BTRST */TByteTAvrDummy::ReadFuseHighBits(){  TByte fusehigh[4] = { 0x58, 0x08, 0, 0 };  Send(fusehigh, 4);  return fusehigh[3];}/* Read Calibration Byte (m163, m323, m128, tn12) */TByteTAvrDummy::ReadCalByte(){  TByte cal[4] = { 0x38, 0, 0, 0 };  Send(cal, 4);  return cal[3];}/* Write Fuse Bits (old):         7     6     5     4     3     2     1     0 2323,8535:                     x     x     x     1     1     1     1     FSTRT 2343:                          x     x     x     1     1     1     1     RCEN 2333,4433:                     x     x     x     BODLV BODEN CKSL2 CKSL1 CKSL0 m103,m603:                     x     x     x     1     EESAV 1     SUT1  SUT0 */voidTAvrDummy::WriteOldFuseBits(TByte val){  TByte oldfuse[4] = { 0xAC, (val & 0x1F) | 0xA0, 0, 0xD2 };  Send(oldfuse, 4);}/* Write Fuse Bits (Low, new):    7     6     5     4     3     2     1     0 m161:                          1     BTRST 1     BODLV BODEN CKSL2 CKSL1 CKSL0 m163,m323:                     BODLV BODEN 1     1     CKSL3 CKSL2 CKSL1 CKSL0 m128:                          BODLV BODEN SUT1  SUT0  CKSL3 CKSL2 CKSL1 CKSL0 tn12:                          BODLV BODEN SPIEN RSTDI CKSL3 CKSL2 CKSL1 CKSL0 tn15:                          BODLV BODEN SPIEN RSTDI 1     1     CKSL1 CKSL0 WARNING (tn12,tn15): writing SPIEN=1 disables further low voltage programming! */voidTAvrDummy::WriteFuseLowBits(TByte val){  TByte fuselow[4] = { 0xAC, 0xA0, 0, val };  Send(fuselow, 4);}/* Write Fuse Bits High:          7     6     5     4     3     2     1     0 m163:                          1     1     1     1     1     BTSZ1 BTSZ0 BTRST m323:                          OCDEN JTGEN 1     1     EESAV BTSZ1 BTSZ0 BTRST m128:                          OCDEN JTGEN x     CKOPT EESAV BTSZ1 BTSZ0 BTRST */voidTAvrDummy::WriteFuseHighBits(TByte val){  TByte fusehigh[4] = { 0xAC, 0xA8, 0, val };  Send(fusehigh, 4);}voidTAvrDummy::WriteByte(TAddr addr, TByte byte, bool flush_buffer){  struct timeval t_start_wr, t_start_poll, t_wait, t_timeout, t_end, t_write;  TByte rbyte=0;  bool device_not_erased=false;    /* Poll data if use_data_polling is enabled and if page mode     is enabled, flash is not selected */  bool poll_data = ((segment==SEG_FLASH && !page_size) || segment==SEG_EEPROM)		   && use_data_polling && TestFeatures(AVR_BYTE_POLL);  CheckMemoryRange(addr);    /* For speed, don't program a byte that is already there     (such as 0xFF after chip erase).  */  if (poll_data){    rbyte=ReadByte(addr);    if (rbyte == byte){return;}    if (rbyte != 0xff){device_not_erased=true;}  }    t_wait.tv_sec = 0;  t_wait.tv_usec = 500000;  gettimeofday(&t_start_wr, NULL);  if (segment==SEG_FLASH){          /* PAGE MODE PROGRAMMING:       If page mode is enabled cache page address.       When current address is out of the page address       flush page buffer and continue programming.    */    if (page_size) {      Info(4, "Loading data to address: %d (page_addr_fetched=%s)\n",         addr, page_addr_fetched?"Yes":"No");	      if (page_addr_fetched && page_addr != (addr & ~(page_size - 1))){        WriteProgramMemoryPage();	      }      if (page_addr_fetched==false){        page_addr=addr & ~(page_size - 1);	page_addr_fetched=true;      }      if (flush_buffer){WriteProgramMemoryPage();}    }    TByte hl = (addr & 1) ? 0x48 : 0x40;    TByte flash [4] = { hl,			(TByte)((addr >> 9) & 0xff),			(TByte)((addr >> 1) & 0xff),			byte };    Send(flash, 4);    /* Remember the last non-0xFF byte written, for page write polling.  */    if (byte != 0xFF) {      page_poll_addr = addr;      page_poll_byte = byte;    }    /* We do not need to wait for each byte in page mode programming */    if (page_size){return;}        t_wait.tv_usec = Get_t_wd_flash();  }  else if (segment==SEG_EEPROM){    TByte eeprom [4] = { 0xC0, 			 (TByte)((addr>>8)&0xff), 			 (TByte)(addr&0xff),			 byte };    Send(eeprom, 4);      t_wait.tv_usec = Get_t_wd_eeprom();      }  else if (segment==SEG_FUSE) {    Info(3, "Write fuse/lock: byte %d = 0x%02X\n",	 (int) addr, (int) byte);    switch (addr) {    case 0:      if (TestFeatures(AVR_FUSE_NEWWR))	WriteFuseLowBits(byte);      else if (TestFeatures(AVR_FUSE_OLDWR))	WriteOldFuseBits(byte);      break;    case 1:      if (TestFeatures(AVR_FUSE_HIGH))	WriteFuseHighBits(byte);      break;    /* calibration byte (addr == 2) is read only */    case 3:      WriteLockBits(byte);      break;    }    t_wait.tv_usec = Get_t_wd_eeprom();  }  gettimeofday(&t_start_poll, NULL);  /* t_timeout = now + t_wd_prog */  timeradd(&t_start_poll, &t_wait, &t_timeout);  do {    /* Data Polling: if the programmed value reads correctly, and       is not equal to any of the possible P1, P2 read back values,       it is done; else wait until tWD_PROG time has elapsed.       The busy loop here is to avoid rounding up the programming       wait time to 10ms timer ticks (for Linux/x86).  Programming       is not really "hard real time" but 10ms instead of ~4ms for       every byte makes it slow.  gettimeofday() reads the 8254       timer registers (or Pentium cycle counter if available),       so it has much better (microsecond) resolution.      */    gettimeofday(&t_end, NULL);    if (poll_data){      if ((byte == (rbyte = ReadByte(addr))) &&          (byte != 0) && (byte != 0x7F) && (byte != 0x80) && (byte != 0xFF)){	break;      }    }  } while (timercmp(&t_end, &t_timeout, <));    /* Write Statistics */  timersub(&t_end, &t_start_wr, &t_write);  /* t_write = t_end - t_start_wr */  if (poll_data) {    float write_time = 1.0e-6 * t_write.tv_usec + t_write.tv_sec;    total_poll_time += write_time;    if (max_poll_time < write_time)      max_poll_time = write_time;    if (min_poll_time > write_time)      min_poll_time = write_time;    total_poll_cnt++;  }  if (poll_data && byte != rbyte){    if (device_not_erased){      Info(0, "Warning: It seems that device is not erased.\n"              "         Erase it with the --erase option.\n");    }    Info(0, "Error: Data polling readback status: write=0x%02x read=0x%02x\n",       byte, rbyte);    throw Error_Device("If device was erased disable polling with the "      "-dno-poll option.");  }}voidTAvrDummy::FlushWriteBuffer(){  if (page_addr_fetched){    WriteProgramMemoryPage();    }}voidTAvrDummy::ChipErase(){  TByte init[4] = { 0xAC, 0x53, 0x00, 0x00 };  TByte chip_erase [4] = { 0xAC, 0x80, 0x00, 0x00 };  Info(1, "Erasing device ...\n");  Send(init, 4);  Send(chip_erase, 4);  Delay_usec(Get_t_wd_erase());  Delay_usec(Get_t_wd_erase());  Delay_usec(9000);  Delay_usec(9000);  PulseReset();  Delay_usec(9000);  Info(1, "Reinitializing device\n");    EnableAvr();}/*   0 = program (clear bit), 1 = leave unchanged   bit 0 = LB1   bit 1 = LB2   bit 2 = BLB01   bit 3 = BLB02   bit 4 = BLB11   bit 5 = BLB12   bit 6 = 1 (reserved)   bit 7 = 1 (reserved) */voidTAvrDummy::WriteLockBits(TByte bits){  /* This handles both old (byte 2, bits 1-2)     and new (byte 4, bits 0-5) devices.  */  TByte lock[4] = { 0xAC, 0xF9 | ((bits << 1) & 0x06), 0xFF, bits };  TByte rbits;  Info(1, "Writing lock bits ...\n");  Send(lock, 4);  Delay_usec(Get_t_wd_erase());  PulseReset();  Info(1, "Reinitializing device\n");  EnableAvr();  rbits = ReadLockBits();  if (rbits & ~bits)    Info(0, "Warning: lock bits write=0x%02X read=0x%02X\n",	 (int) bits, (int) rbits);}TByteTAvrDummy::ReadLockBits(){  TByte rbits = 0xFF;  if (TestFeatures(AVR_LOCK_BOOT)) {    /* x x BLB12 BLB11 BLB02 BLB01 LB2 LB1 */    rbits = ReadLockFuseBits();  } else if (TestFeatures(AVR_LOCK_RD76)) {    rbits = ReadLockFuseBits();    /* LB1 LB2 x x x x x x -> 1 1 1 1 1 1 LB2 LB1 */    rbits = ((rbits >> 7) & 1) | ((rbits >> 5) & 1) | 0xFC;  } else if (TestFeatures(AVR_LOCK_RD12)) {    rbits = ReadLockFuseBits();    /* x x x x x LB2 LB1 x -> 1 1 1 1 1 1 LB2 LB1 */    rbits = ((rbits >> 1) & 3) | 0xFC;  } else if (GetPartInfo(0) == 0 &&	     GetPartInfo(1) == 1 &&	     GetPartInfo(2) == 2) {    rbits = 0xFC;  }  return rbits;}unsigned intTAvrDummy::GetPollCount(){  return total_poll_cnt;}floatTAvrDummy::GetMinPollTime(){  return min_poll_time;}floatTAvrDummy::GetMaxPollTime(){  return max_poll_time;}floatTAvrDummy::GetTotPollTime(){  return total_poll_time;}voidTAvrDummy::ResetMinMax(){  min_poll_time = 1.0;  max_poll_time = 0.0;  total_poll_time = 0.0;  total_poll_cnt = 0;}/* Constructor*/TAvrDummy::TAvrDummy():  use_data_polling(true){  ResetMinMax();    /* Device Command line options ... */   if (GetCmdParam("-dno-poll", false)){use_data_polling=false;}     EnableAvr();}#endif/* eof */

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -