📄 blcommon.c
字号:
if (!memcmp (hdr, "X000FF\x0A", BL_HDRSIG_SIZE)) {
// determine number of BIN files to be downloaded.
if (!OEMReadData (sizeof (DWORD), (LPBYTE) &dwRecChk)) {
EdbgOutputDebugString("\r\nUnable to read BIN region checksum.\r\n");
HALT (BLERR_MAGIC);
return FALSE;
}
// read BIN region descriptions (start address and length).
if (!OEMReadData (sizeof (DWORD), (LPBYTE) &g_MultiBINInfo.dwNumRegions)
|| !OEMReadData ((g_MultiBINInfo.dwNumRegions * sizeof(RegionInfo)), (LPBYTE) &g_MultiBINInfo.Region[0])) {
EdbgOutputDebugString("\r\nUnable to read BIN region descriptors.\r\n");
HALT (BLERR_MAGIC);
return FALSE;
}
// verify the packet checksum.
if (!VerifyChecksum((g_MultiBINInfo.dwNumRegions * sizeof(RegionInfo)), (LPBYTE) &g_MultiBINInfo.Region, dwRecChk)) {
EdbgOutputDebugString ("\r\nBIN region descriptor packet failed checksum.\r\n");
HALT (BLERR_CHECKSUM);
return FALSE;
}
// provide the region information to the OEM.
if (g_pOEMMultiBINNotify) {
g_pOEMMultiBINNotify((const PMultiBINInfo)&g_MultiBINInfo);
}
// look for next download...
nNumRegions = (BYTE)(g_MultiBINInfo.dwNumRegions + 1); // +1 to account for this packet.
continue;
}
else {
// make sure it is a standard BIN file.
if (memcmp (hdr, "B000FF\x0A", BL_HDRSIG_SIZE)) {
EdbgOutputDebugString ("\r\nThis is not a .BIN file %x %x %x %x %x %x %x\r\n",
hdr[0], hdr[1], hdr[2], hdr[3], hdr[4], hdr[5], hdr[6]);
HALT (BLERR_MAGIC);
return FALSE;
}
}
// read image start/length
if (!OEMReadData (sizeof (DWORD), (LPBYTE) &dwImageStart)
|| !OEMReadData (sizeof (DWORD), (LPBYTE) &dwImageLength)) {
EdbgOutputDebugString ("Unable to read image start/length\r\n");
HALT (BLERR_MAGIC);
return FALSE;
}
// if this is a single-bin download, record the bin file information and notify the OEM.
if (!g_MultiBINInfo.dwNumRegions) {
g_MultiBINInfo.dwNumRegions = 1;
g_MultiBINInfo.Region[0].dwRegionStart = dwImageStart;
g_MultiBINInfo.Region[0].dwRegionLength = dwImageLength;
// provide the region information to the OEM.
if (g_pOEMMultiBINNotify) {
g_pOEMMultiBINNotify((const PMultiBINInfo)&g_MultiBINInfo);
}
}
// give the OEM a chance to verify memory
if (g_pOEMVerifyMemory && !g_pOEMVerifyMemory (dwImageStart, dwImageLength)) {
EdbgOutputDebugString ("!OEMVERIFYMEMORY: Invalid image\r\n");
HALT (BLERR_OEMVERIFY);
return FALSE;
}
// check for flash image. Start erasing if it is.
if ((fIsFlash = OEMIsFlashAddr (dwImageStart))
&& !OEMStartEraseFlash (dwImageStart, dwImageLength)) {
EdbgOutputDebugString ("Invalid Flash Address/Length\r\n");
HALT (BLERR_FLASHADDR);
return FALSE;
}
// read records (start with address, length, and checksum)
while (OEMReadData (sizeof (DWORD), (LPBYTE) &dwRecAddr)
&& OEMReadData (sizeof (DWORD), (LPBYTE) &dwRecLen)
&& OEMReadData (sizeof (DWORD), (LPBYTE) &dwRecChk)) {
RETAILMSG(1, (_T("DownloadImage RecAddr: 0x%x\r\n"), dwRecAddr));
RETAILMSG(1, (_T("DownloadImage RecLen: 0x%x\r\n"), dwRecLen));
RETAILMSG(1, (_T("DownloadImage RecChk: 0x%x\r\n"), dwRecChk));
// check for last record
if (!dwRecAddr && !dwRecChk) {
// if this is the kernel region, update launch address
if (IsKernelRegion(dwImageStart, dwImageLength)) {
*pdwImageStart = dwImageStart;
*pdwImageLength = dwImageLength;
*pdwLaunchAddr = dwRecLen;
RETAILMSG(1, (_T("dwImageStart : 0x%x\r\n"), dwImageStart));
RETAILMSG(1, (_T("dwImageLength: 0x%x\r\n"), dwImageLength));
RETAILMSG(1, (_T("LaunchAddr : 0x%x\r\n"), dwRecLen));
}
// write to flash if it's flash image
if (fIsFlash) {
// finish the flash erase
if (!OEMFinishEraseFlash ()) {
HALT (BLERR_FLASH_ERASE);
return FALSE;
}
// Before writing the image to flash, optionally check the image signature.
if (g_pOEMCheckSignature)
{
if (!g_pOEMCheckSignature(dwImageStart, g_dwROMOffset, *pdwLaunchAddr, TRUE))
HALT(BLERR_WHQL_SIGNATURE);
}
}
// On to the next (possible) BIN file...
break;
}
// map the record address (FLASH data is cached, for example)
lpDest = OEMMapMemAddr (dwImageStart, dwRecAddr);
// read data block
if (!OEMReadData (dwRecLen, lpDest)) {
EdbgOutputDebugString ("****** Data record %d corrupted, ABORT!!! ******\r\n", nPkgNum);
HALT (BLERR_CORRUPTED_DATA);
return FALSE;
}
if (!VerifyChecksum (dwRecLen, lpDest, dwRecChk)) {
EdbgOutputDebugString ("****** Checksum failure on record %d, ABORT!!! ******\r\n", nPkgNum);
HALT (BLERR_CHECKSUM);
return FALSE;
}
// Look for ROMHDR to compute ROM offset. NOTE: romimage guarantees that the record containing
// the TOC signature and pointer will always come before the record that contains the ROMHDR contents.
//
if (dwRecLen == sizeof(ROMHDR) && (*(LPDWORD) OEMMapMemAddr(dwImageStart, dwImageStart + ROM_SIGNATURE_OFFSET) == ROM_SIGNATURE))
{
DWORD dwTempOffset = (dwRecAddr - *(LPDWORD)OEMMapMemAddr(dwImageStart, dwImageStart + ROM_SIGNATURE_OFFSET + sizeof(ULONG)));
ROMHDR *pROMHdr = (ROMHDR *)lpDest;
// Check to make sure this record really contains the ROMHDR.
//
if ((pROMHdr->physfirst == (dwImageStart - dwTempOffset)) &&
(pROMHdr->physlast == (dwImageStart - dwTempOffset + dwImageLength)) &&
(DWORD)(HIWORD(pROMHdr->dllfirst << 16) <= pROMHdr->dlllast) &&
(DWORD)(LOWORD(pROMHdr->dllfirst << 16) <= pROMHdr->dlllast))
{
g_dwROMOffset = dwTempOffset;
EdbgOutputDebugString("rom_offset=0x%x.\r\n", g_dwROMOffset);
}
}
// verify partial checksum
OEMShowProgress (nPkgNum ++);
if (fIsFlash) {
OEMContinueEraseFlash ();
}
}
}
while (--nNumRegions);
if (fIsFlash) {
nNumRegions = (BYTE)g_MultiBINInfo.dwNumRegions;
while (nNumRegions--) {
if (!OEMWriteFlash (g_MultiBINInfo.Region[nNumRegions].dwRegionStart, g_MultiBINInfo.Region[nNumRegions].dwRegionLength)) {
HALT (BLERR_FLASH_WRITE);
return FALSE;
}
}
}
return TRUE;
}
/*
@func BOOLEAN | IsKernelRegion | Determines if the expanded BIN file provided contains the kernel image.
@rdesc TRUE if the region contains the kernel image, FALSE if it doesn't.
@comm <l Download Image>
@xref
@notes dwCurrentBase is the base address where the BIN records are currently stored (this can be a RAM, a RAM
file cache, or flash). dwImageStoreBase is the images base storage address (this is the base address of potentially
multiple BIN regions and can be in RAM or flash) and is used to translate addresses to the file cache area.
dwROMOffset is the difference between the address where the BIN records are stored versus where they're fixed-up
to run from (for example, an image may be stored in flash, but fixed-up to run in RAM).
*/
static BOOL IsKernelRegion(DWORD dwRegionStart, DWORD dwRegionLength)
{
DWORD dwCacheAddress = 0;
ROMHDR *pROMHeader;
DWORD dwNumModules = 0;
TOCentry *pTOC;
if (dwRegionStart == 0 || dwRegionLength == 0)
return(FALSE);
if (*(LPDWORD) OEMMapMemAddr (dwRegionStart, dwRegionStart + ROM_SIGNATURE_OFFSET) != ROM_SIGNATURE)
return FALSE;
// A pointer to the ROMHDR structure lives just past the ROM_SIGNATURE (which is a longword value). Note that
// this pointer is remapped since it might be a flash address (image destined for flash), but is actually cached
// in RAM.
//
dwCacheAddress = *(LPDWORD) OEMMapMemAddr (dwRegionStart, dwRegionStart + ROM_SIGNATURE_OFFSET + sizeof(ULONG));
pROMHeader = (ROMHDR *) OEMMapMemAddr (dwRegionStart, dwCacheAddress);
// Make sure sure are some modules in the table of contents.
//
if ((dwNumModules = pROMHeader->nummods) == 0)
return FALSE;
// Locate the table of contents and search for the kernel executable and the TOC immediately follows the ROMHDR.
//
pTOC = (TOCentry *)(pROMHeader + 1);
while(dwNumModules--) {
LPBYTE pFileName = OEMMapMemAddr(dwRegionStart, (DWORD)pTOC->lpszFileName);
if (!strcmp(pFileName, "nk.exe")) {
return TRUE;
}
++pTOC;
}
return FALSE;
}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -