📄 skvpd.c
字号:
/****************************************************************************** * * Name: skvpd.c * Project: GEnesis, PCI Gigabit Ethernet Adapter * Version: $Revision: 1.37 $ * Date: $Date: 2003/01/13 10:42:45 $ * Purpose: Shared software to read and write VPD data * ******************************************************************************//****************************************************************************** * * (C)Copyright 1998-2003 SysKonnect GmbH. * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * The information in this file is provided "AS IS" without warranty. * ******************************************************************************//* Please refer skvpd.txt for infomation how to include this module */static const char SysKonnectFileId[] = "@(#)$Id: skvpd.c,v 1.37 2003/01/13 10:42:45 rschmidt Exp $ (C) SK";#include "h/skdrv1st.h"#include "h/sktypes.h"#include "h/skdebug.h"#include "h/skdrv2nd.h"/* * Static functions */#ifndef SK_KR_PROTOstatic SK_VPD_PARA *vpd_find_para( SK_AC *pAC, const char *key, SK_VPD_PARA *p);#else /* SK_KR_PROTO */static SK_VPD_PARA *vpd_find_para();#endif /* SK_KR_PROTO *//* * waits for a completion of a VPD transfer * The VPD transfer must complete within SK_TICKS_PER_SEC/16 * * returns 0: success, transfer completes * error exit(9) with a error message */static int VpdWait(SK_AC *pAC, /* Adapters context */SK_IOC IoC, /* IO Context */int event) /* event to wait for (VPD_READ / VPD_write) completion*/{ SK_U64 start_time; SK_U16 state; SK_DBG_MSG(pAC,SK_DBGMOD_VPD, SK_DBGCAT_CTRL, ("VPD wait for %s\n", event?"Write":"Read")); start_time = SkOsGetTime(pAC); do { if (SkOsGetTime(pAC) - start_time > SK_TICKS_PER_SEC) { /* Bug fix AF: Thu Mar 28 2002 * Do not call: VPD_STOP(pAC, IoC); * A pending VPD read cycle can not be aborted by writing * VPD_WRITE to the PCI_VPD_ADR_REG (VPD address register). * Although the write threshold in the OUR-register protects * VPD read only space from being overwritten this does not * protect a VPD read from being `converted` into a VPD write * operation (on the fly). As a consequence the VPD_STOP would * delete VPD read only data. In case of any problems with the * I2C bus we exit the loop here. The I2C read operation can * not be aborted except by a reset (->LR). */ SK_DBG_MSG(pAC, SK_DBGMOD_VPD, SK_DBGCAT_FATAL | SK_DBGCAT_ERR, ("ERROR:VPD wait timeout\n")); return(1); } VPD_IN16(pAC, IoC, PCI_VPD_ADR_REG, &state); SK_DBG_MSG(pAC, SK_DBGMOD_VPD, SK_DBGCAT_CTRL, ("state = %x, event %x\n",state,event)); } while((int)(state & PCI_VPD_FLAG) == event); return(0);}#ifdef SKDIAG/* * Read the dword at address 'addr' from the VPD EEPROM. * * Needed Time: MIN 1,3 ms MAX 2,6 ms * * Note: The DWord is returned in the endianess of the machine the routine * is running on. * * Returns the data read. */SK_U32 VpdReadDWord(SK_AC *pAC, /* Adapters context */SK_IOC IoC, /* IO Context */int addr) /* VPD address */{ SK_U32 Rtv; /* start VPD read */ SK_DBG_MSG(pAC, SK_DBGMOD_VPD, SK_DBGCAT_CTRL, ("VPD read dword at 0x%x\n",addr)); addr &= ~VPD_WRITE; /* ensure the R/W bit is set to read */ VPD_OUT16(pAC, IoC, PCI_VPD_ADR_REG, (SK_U16)addr); /* ignore return code here */ (void)VpdWait(pAC, IoC, VPD_READ); /* Don't swap here, it's a data stream of bytes */ Rtv = 0; VPD_IN32(pAC, IoC, PCI_VPD_DAT_REG, &Rtv); SK_DBG_MSG(pAC, SK_DBGMOD_VPD, SK_DBGCAT_CTRL, ("VPD read dword data = 0x%x\n",Rtv)); return(Rtv);}#endif /* SKDIAG */#if 0/* Write the dword 'data' at address 'addr' into the VPD EEPROM, and verify that the data is written. Needed Time:. MIN MAX. -------------------------------------------------------------------. write 1.8 ms 3.6 ms. internal write cyles 0.7 ms 7.0 ms. -------------------------------------------------------------------. over all program time 2.5 ms 10.6 ms. read 1.3 ms 2.6 ms. -------------------------------------------------------------------. over all 3.8 ms 13.2 ms. Returns 0: success 1: error, I2C transfer does not terminate 2: error, data verify error */static int VpdWriteDWord(SK_AC *pAC, /* pAC pointer */SK_IOC IoC, /* IO Context */int addr, /* VPD address */SK_U32 data) /* VPD data to write */{ /* start VPD write */ /* Don't swap here, it's a data stream of bytes */ SK_DBG_MSG(pAC, SK_DBGMOD_VPD, SK_DBGCAT_CTRL, ("VPD write dword at addr 0x%x, data = 0x%x\n",addr,data)); VPD_OUT32(pAC, IoC, PCI_VPD_DAT_REG, (SK_U32)data); /* But do it here */ addr |= VPD_WRITE; VPD_OUT16(pAC, IoC, PCI_VPD_ADR_REG, (SK_U16)(addr | VPD_WRITE)); /* this may take up to 10,6 ms */ if (VpdWait(pAC, IoC, VPD_WRITE)) { SK_DBG_MSG(pAC, SK_DBGMOD_VPD, SK_DBGCAT_ERR, ("Write Timed Out\n")); return(1); }; /* verify data */ if (VpdReadDWord(pAC, IoC, addr) != data) { SK_DBG_MSG(pAC, SK_DBGMOD_VPD, SK_DBGCAT_ERR | SK_DBGCAT_FATAL, ("Data Verify Error\n")); return(2); } return(0);} /* VpdWriteDWord */#endif /* 0 *//* * Read one Stream of 'len' bytes of VPD data, starting at 'addr' from * or to the I2C EEPROM. * * Returns number of bytes read / written. */static int VpdWriteStream(SK_AC *pAC, /* Adapters context */SK_IOC IoC, /* IO Context */char *buf, /* data buffer */int Addr, /* VPD start address */int Len) /* number of bytes to read / to write */{ int i; int j; SK_U16 AdrReg; int Rtv; SK_U8 * pComp; /* Compare pointer */ SK_U8 Data; /* Input Data for Compare */ /* Init Compare Pointer */ pComp = (SK_U8 *) buf; for (i = 0; i < Len; i++, buf++) { if ((i%sizeof(SK_U32)) == 0) { /* * At the begin of each cycle read the Data Reg * So it is initialized even if only a few bytes * are written. */ AdrReg = (SK_U16) Addr; AdrReg &= ~VPD_WRITE; /* READ operation */ VPD_OUT16(pAC, IoC, PCI_VPD_ADR_REG, AdrReg); /* Wait for termination */ Rtv = VpdWait(pAC, IoC, VPD_READ); if (Rtv != 0) { return(i); } } /* Write current Byte */ VPD_OUT8(pAC, IoC, PCI_VPD_DAT_REG + (i%sizeof(SK_U32)), *(SK_U8*)buf); if (((i%sizeof(SK_U32)) == 3) || (i == (Len - 1))) { /* New Address needs to be written to VPD_ADDR reg */ AdrReg = (SK_U16) Addr; Addr += sizeof(SK_U32); AdrReg |= VPD_WRITE; /* WRITE operation */ VPD_OUT16(pAC, IoC, PCI_VPD_ADR_REG, AdrReg); /* Wait for termination */ Rtv = VpdWait(pAC, IoC, VPD_WRITE); if (Rtv != 0) { SK_DBG_MSG(pAC, SK_DBGMOD_VPD, SK_DBGCAT_ERR, ("Write Timed Out\n")); return(i - (i%sizeof(SK_U32))); } /* * Now re-read to verify */ AdrReg &= ~VPD_WRITE; /* READ operation */ VPD_OUT16(pAC, IoC, PCI_VPD_ADR_REG, AdrReg); /* Wait for termination */ Rtv = VpdWait(pAC, IoC, VPD_READ); if (Rtv != 0) { SK_DBG_MSG(pAC, SK_DBGMOD_VPD, SK_DBGCAT_ERR, ("Verify Timed Out\n")); return(i - (i%sizeof(SK_U32))); } for (j = 0; j <= (int)(i%sizeof(SK_U32)); j++, pComp++) { VPD_IN8(pAC, IoC, PCI_VPD_DAT_REG + j, &Data); if (Data != *pComp) { /* Verify Error */ SK_DBG_MSG(pAC, SK_DBGMOD_VPD, SK_DBGCAT_ERR, ("WriteStream Verify Error\n")); return(i - (i%sizeof(SK_U32)) + j); } } } } return(Len);} /* * Read one Stream of 'len' bytes of VPD data, starting at 'addr' from * or to the I2C EEPROM. * * Returns number of bytes read / written. */static int VpdReadStream(SK_AC *pAC, /* Adapters context */SK_IOC IoC, /* IO Context */char *buf, /* data buffer */int Addr, /* VPD start address */int Len) /* number of bytes to read / to write */{ int i; SK_U16 AdrReg; int Rtv; for (i = 0; i < Len; i++, buf++) { if ((i%sizeof(SK_U32)) == 0) { /* New Address needs to be written to VPD_ADDR reg */ AdrReg = (SK_U16) Addr; Addr += sizeof(SK_U32); AdrReg &= ~VPD_WRITE; /* READ operation */ VPD_OUT16(pAC, IoC, PCI_VPD_ADR_REG, AdrReg); /* Wait for termination */ Rtv = VpdWait(pAC, IoC, VPD_READ); if (Rtv != 0) { return(i); } } VPD_IN8(pAC, IoC, PCI_VPD_DAT_REG + (i%sizeof(SK_U32)), (SK_U8 *)buf); } return(Len);}/* * Read ore writes 'len' bytes of VPD data, starting at 'addr' from * or to the I2C EEPROM. * * Returns number of bytes read / written. */static int VpdTransferBlock(SK_AC *pAC, /* Adapters context */SK_IOC IoC, /* IO Context */char *buf, /* data buffer */int addr, /* VPD start address */int len, /* number of bytes to read / to write */int dir) /* transfer direction may be VPD_READ or VPD_WRITE */{ int Rtv; /* Return value */ int vpd_rom_size; SK_DBG_MSG(pAC, SK_DBGMOD_VPD, SK_DBGCAT_CTRL, ("VPD %s block, addr = 0x%x, len = %d\n", dir ? "write" : "read", addr, len)); if (len == 0) return(0); vpd_rom_size = pAC->vpd.rom_size; if (addr > vpd_rom_size - 4) { SK_DBG_MSG(pAC, SK_DBGMOD_VPD, SK_DBGCAT_ERR | SK_DBGCAT_FATAL, ("Address error: 0x%x, exp. < 0x%x\n", addr, vpd_rom_size - 4)); return(0); } if (addr + len > vpd_rom_size) { len = vpd_rom_size - addr; SK_DBG_MSG(pAC, SK_DBGMOD_VPD, SK_DBGCAT_ERR, ("Warning: len was cut to %d\n", len)); } if (dir == VPD_READ) { Rtv = VpdReadStream(pAC, IoC, buf, addr, len); } else { Rtv = VpdWriteStream(pAC, IoC, buf, addr, len); } return(Rtv);}#ifdef SKDIAG/* * Read 'len' bytes of VPD data, starting at 'addr'. * * Returns number of bytes read. */int VpdReadBlock(SK_AC *pAC, /* pAC pointer */SK_IOC IoC, /* IO Context */char *buf, /* buffer were the data should be stored */int addr, /* start reading at the VPD address */int len) /* number of bytes to read */{ return(VpdTransferBlock(pAC, IoC, buf, addr, len, VPD_READ));}/* * Write 'len' bytes of *but to the VPD EEPROM, starting at 'addr'. * * Returns number of bytes writes. */int VpdWriteBlock(SK_AC *pAC, /* pAC pointer */SK_IOC IoC, /* IO Context */char *buf, /* buffer, holds the data to write */int addr, /* start writing at the VPD address */int len) /* number of bytes to write */{ return(VpdTransferBlock(pAC, IoC, buf, addr, len, VPD_WRITE));}#endif /* SKDIAG *//* * (re)initialize the VPD buffer * * Reads the VPD data from the EEPROM into the VPD buffer. * Get the remaining read only and read / write space. * * return 0: success * 1: fatal VPD error */static int VpdInit(SK_AC *pAC, /* Adapters context */SK_IOC IoC) /* IO Context */{ SK_VPD_PARA *r, rp; /* RW or RV */ int i; unsigned char x; int vpd_size; SK_U16 dev_id; SK_U32 our_reg2; SK_DBG_MSG(pAC, SK_DBGMOD_VPD, SK_DBGCAT_INIT, ("VpdInit .. ")); VPD_IN16(pAC, IoC, PCI_DEVICE_ID, &dev_id); VPD_IN32(pAC, IoC, PCI_OUR_REG_2, &our_reg2); pAC->vpd.rom_size = 256 << ((our_reg2 & PCI_VPD_ROM_SZ) >> 14); /* * this function might get used before the hardware is initialized * therefore we cannot always trust in GIChipId */ if (((pAC->vpd.v.vpd_status & VPD_VALID) == 0 && dev_id != VPD_DEV_ID_GENESIS) || ((pAC->vpd.v.vpd_status & VPD_VALID) != 0 && !pAC->GIni.GIGenesis)) { /* for Yukon the VPD size is always 256 */ vpd_size = VPD_SIZE_YUKON; } else { /* Genesis uses the maximum ROM size up to 512 for VPD */ if (pAC->vpd.rom_size > VPD_SIZE_GENESIS) { vpd_size = VPD_SIZE_GENESIS; } else { vpd_size = pAC->vpd.rom_size; } } /* read the VPD data into the VPD buffer */ if (VpdTransferBlock(pAC, IoC, pAC->vpd.vpd_buf, 0, vpd_size, VPD_READ) != vpd_size) { SK_DBG_MSG(pAC, SK_DBGMOD_VPD, SK_DBGCAT_ERR, ("Block Read Error\n")); return(1); } pAC->vpd.vpd_size = vpd_size; /* Asus K8V Se Deluxe bugfix. Correct VPD content */ /* MBo April 2004 */ if (((unsigned char)pAC->vpd.vpd_buf[0x3f] == 0x38) && ((unsigned char)pAC->vpd.vpd_buf[0x40] == 0x3c) && ((unsigned char)pAC->vpd.vpd_buf[0x41] == 0x45)) { printk("sk98lin: Asus mainboard with buggy VPD? " "Correcting data.\n"); pAC->vpd.vpd_buf[0x40] = 0x38; } /* find the end tag of the RO area */ if (!(r = vpd_find_para(pAC, VPD_RV, &rp))) { SK_DBG_MSG(pAC, SK_DBGMOD_VPD, SK_DBGCAT_ERR | SK_DBGCAT_FATAL, ("Encoding Error: RV Tag not found\n")); return(1); } if (r->p_val + r->p_len > pAC->vpd.vpd_buf + vpd_size/2) { SK_DBG_MSG(pAC,SK_DBGMOD_VPD,SK_DBGCAT_ERR | SK_DBGCAT_FATAL, ("Encoding Error: Invalid VPD struct size\n")); return(1); } pAC->vpd.v.vpd_free_ro = r->p_len - 1; /* test the checksum */ for (i = 0, x = 0; (unsigned)i <= (unsigned)vpd_size/2 - r->p_len; i++) { x += pAC->vpd.vpd_buf[i]; } if (x != 0) { /* checksum error */ SK_DBG_MSG(pAC, SK_DBGMOD_VPD, SK_DBGCAT_ERR | SK_DBGCAT_FATAL, ("VPD Checksum Error\n")); return(1); } /* find and check the end tag of the RW area */ if (!(r = vpd_find_para(pAC, VPD_RW, &rp))) { SK_DBG_MSG(pAC, SK_DBGMOD_VPD, SK_DBGCAT_ERR | SK_DBGCAT_FATAL, ("Encoding Error: RV Tag not found\n")); return(1); } if (r->p_val < pAC->vpd.vpd_buf + vpd_size/2) { SK_DBG_MSG(pAC, SK_DBGMOD_VPD, SK_DBGCAT_ERR | SK_DBGCAT_FATAL, ("Encoding Error: Invalid VPD struct size\n")); return(1); } pAC->vpd.v.vpd_free_rw = r->p_len; /* everything seems to be ok */ if (pAC->GIni.GIChipId != 0) { pAC->vpd.v.vpd_status |= VPD_VALID; } SK_DBG_MSG(pAC, SK_DBGMOD_VPD, SK_DBGCAT_INIT, ("done. Free RO = %d, Free RW = %d\n", pAC->vpd.v.vpd_free_ro, pAC->vpd.v.vpd_free_rw)); return(0);}/* * find the Keyword 'key' in the VPD buffer and fills the * parameter struct 'p' with it's values * * returns *p success * 0: parameter was not found or VPD encoding error */static SK_VPD_PARA *vpd_find_para(SK_AC *pAC, /* common data base */const char *key, /* keyword to find (e.g. "MN") */SK_VPD_PARA *p) /* parameter description struct */{ char *v ; /* points to VPD buffer */ int max; /* Maximum Number of Iterations */ v = pAC->vpd.vpd_buf; max = 128; SK_DBG_MSG(pAC, SK_DBGMOD_VPD, SK_DBGCAT_CTRL, ("VPD find para %s .. ",key)); /* check mandatory resource type ID string (Product Name) */ if (*v != (char)RES_ID) { SK_DBG_MSG(pAC, SK_DBGMOD_VPD, SK_DBGCAT_ERR | SK_DBGCAT_FATAL, ("Error: 0x%x missing\n", RES_ID)); return NULL; } if (strcmp(key, VPD_NAME) == 0) { p->p_len = VPD_GET_RES_LEN(v); p->p_val = VPD_GET_VAL(v); SK_DBG_MSG(pAC, SK_DBGMOD_VPD, SK_DBGCAT_CTRL, ("found, len = %d\n", p->p_len)); return(p); } v += 3 + VPD_GET_RES_LEN(v) + 3; for (;; ) { if (SK_MEMCMP(key,v,2) == 0) { p->p_len = VPD_GET_VPD_LEN(v); p->p_val = VPD_GET_VAL(v); SK_DBG_MSG(pAC, SK_DBGMOD_VPD, SK_DBGCAT_CTRL, ("found, len = %d\n",p->p_len)); return(p); } /* exit when reaching the "RW" Tag or the maximum of itera. */ max--; if (SK_MEMCMP(VPD_RW,v,2) == 0 || max == 0) { break; } if (SK_MEMCMP(VPD_RV,v,2) == 0) { v += 3 + VPD_GET_VPD_LEN(v) + 3; /* skip VPD-W */ } else { v += 3 + VPD_GET_VPD_LEN(v); } SK_DBG_MSG(pAC, SK_DBGMOD_VPD, SK_DBGCAT_CTRL, ("scanning '%c%c' len = %d\n",v[0],v[1],v[2])); }#ifdef DEBUG SK_DBG_MSG(pAC, SK_DBGMOD_VPD, SK_DBGCAT_CTRL, ("not found\n")); if (max == 0) { SK_DBG_MSG(pAC, SK_DBGMOD_VPD, SK_DBGCAT_ERR | SK_DBGCAT_FATAL,
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -