📄 kitleth.c
字号:
//
// Copyright (c) Microsoft Corporation. All rights reserved.
//
//
// Use of this source code is subject to the terms of the Microsoft end-user
// license agreement (EULA) under which you licensed this SOFTWARE PRODUCT.
// If you did not accept the terms of the EULA, you are not authorized to use
// this source code. For a copy of the EULA, please see the LICENSE.RTF on your
// install media.
//
// -----------------------------------------------------------------------------
//
// THIS CODE AND INFORMATION IS PROVIDED "AS IS" WITHOUT WARRANTY OF
// ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING BUT NOT LIMITED TO
// THE IMPLIED WARRANTIES OF MERCHANTABILITY AND/OR FITNESS FOR A
// PARTICULAR PURPOSE.
//
// -----------------------------------------------------------------------------
#include <windows.h>
#include <x86kitl.h>
#include <pci.h>
static KTIL_NIC_INFO gKitlNic;
PCKITL_NIC_INFO g_pKitlPCINic; // non-null only if it's PCI. Used in RegisterKitl
PCSUPPORTED_NIC FindNICByType (UCHAR ucType)
{
int i;
for (i = 0; i < g_nNumNicSupported; i ++) {
if (g_NicSupported[i].Type == ucType) {
return &g_NicSupported[i];
}
}
return NULL;
}
LPCSTR FindNICAbbrev (DWORD dwUpperMAC)
{
int i;
for (i = 0; i < g_nNumNicSupported; i ++) {
if (g_NicSupported[i].dwUpperMAC == dwUpperMAC) {
return g_NicSupported[i].szAbbrev;
}
}
return "";
}
PCSUPPORTED_NIC FindNICByVendor (PCI_COMMON_CONFIG *pPciCfg)
{
int i;
for (i = 0; i < g_nNumNicSupported; i ++) {
if ((g_NicSupported[i].wVenId == pPciCfg->VendorID) &&
(g_NicSupported[i].wDevId == pPciCfg->DeviceID)) {
return &g_NicSupported[i];
}
}
return NULL;
}
PCKITL_NIC_INFO InitKitlNIC (DWORD dwIrq, DWORD dwIoBase, DWORD dwDfltType)
{
PCI_COMMON_CONFIG pciConfig;
int bus, device, function;
PCSUPPORTED_NIC pNicFound;
int length = 0;
KITLOutputDebugString("InitKitlNIC: Searching for PCI Ethernet NIC (dwIrq = %x, dwIoBase = %x, dwDfltType = %x) ...\r\n",
dwIrq, dwIoBase, dwDfltType);
// iterate through buses
for (bus = 0; bus < PCI_MAX_BUS; bus++) {
// iterate through devices
for (device = 0; device < PCI_MAX_DEVICES; device++) {
// iterate through functions
for (function = 0; function < PCI_MAX_FUNCTION; function++) {
// read PCI config data
length = PCIReadBusData ( bus,
device,
function,
&pciConfig,
0,
(sizeof(pciConfig) - sizeof(pciConfig.DeviceSpecific)));
if (length == 0 || (pciConfig.VendorID == 0xFFFF))
break;
// network controller or USB?
if ( ( (pciConfig.BaseClass == PCI_CLASS_NETWORK_CTLR)
&& (pciConfig.SubClass == PCI_SUBCLASS_NET_ETHERNET_CTLR)) // Network device.
|| ( (pciConfig.BaseClass == PCI_CLASS_BRIDGE_DEV)
&& (pciConfig.SubClass == PCI_SUBCLASS_BR_OTHER))) { // Other Unknown Special Devices
DWORD dwFoundBase = pciConfig.u.type0.BaseAddresses[0] & 0xFFFFFFFC;
DWORD dwFoundIrq = pciConfig.u.type0.InterruptLine;
if (dwFoundIrq && dwFoundBase) {
if (!dwIrq // IRQ not specified -- use 1st found
|| (!dwIoBase && (dwIrq == dwFoundIrq)) // IRQ match, no IO base specified
|| ((dwIoBase == dwFoundBase) && (dwIrq == dwFoundIrq))) { // both IRQ and IOBase match
pNicFound = FindNICByVendor (&pciConfig);
if (!pNicFound) {
// not in the vendor list, see if the type is supported
pNicFound = FindNICByType ((UCHAR) dwDfltType);
}
if (pNicFound) {
// found NIC card
gKitlNic.dwIoBase = dwFoundBase;
gKitlNic.dwIrq = dwFoundIrq;
gKitlNic.dwBus = bus;
gKitlNic.dwDevice = device;
gKitlNic.dwFunction = function;
gKitlNic.pDriver = pNicFound->pDriver;
gKitlNic.dwType = pNicFound->Type;
memcpy (&gKitlNic.pciConfig, &pciConfig, sizeof(pciConfig));
KITLOutputDebugString ("InitKitlNIC: Found PCI Ethernet NIC (type = %x, IRQ=%d, IOBase=0x%x).\r\n",
pNicFound->Type, dwFoundIrq, dwFoundBase);
return g_pKitlPCINic = &gKitlNic;
}
}
}
}
if (function == 0 && !(pciConfig.HeaderType & 0x80))
break;
}
if (length == 0)
break;
}
if (length == 0 && device == 0)
break;
}
// can't find it on PCI bus, if IRQ and IoBase are specified, use it
if (dwIrq && dwIoBase && (pNicFound = FindNICByType ((UCHAR) dwDfltType))) {
gKitlNic.dwIoBase = dwIoBase;
gKitlNic.dwIrq = dwIrq;
gKitlNic.pDriver = pNicFound->pDriver;
gKitlNic.dwType = dwDfltType;
KITLOutputDebugString ("InitKitlNIC: Can't find PCI Ethernet NIC, use specified data (type = %x, IRQ=%d, IOBase=0x%x), use .\r\n",
pNicFound->Type, dwIrq, dwIoBase);
return &gKitlNic;
}
return NULL;
}
//------------------------------------------------------------------------------
//------------------------------------------------------------------------------
static void
itoa10(
int n,
CHAR s[],
int bufflen
)
{
int i = 0;
// Get absolute value of number
unsigned int val = (unsigned int)((n < 0) ? -n : n);
// Extract digits in reverse order
while (val)
{
// Make sure we don't step off the end of the character array (leave
// room for the possible '-' sign and the null terminator).
if (i < (bufflen - 2))
{
s[i++] = (val % 10) + '0';
}
val /= 10;
}
// Add sign if number negative
if (n < 0) s[i++] = '-';
s[i--] = '\0';
// Reverse string
for (n = 0; n < i; n++, i--) {
char swap = s[n];
s[n] = s[i];
s[i] = swap;
}
}
static DWORD UpperDWFromMAC (UINT16 wMAC [])
{
DWORD ret;
//
// The WORDs in wMAC field are in net order, so we need to do some
// serious shifting around.
// A hex ethernet address of 12 34 56 78 9a bc is stored in wMAC array as
// wMAC[0] = 3412, wMAC[1] = 7856, wMAC[2] = bc9a.
// The 4 byte return value should look like 0x00123456
//
ret = (wMAC[0] & 0x00ff) << 16;
ret |= wMAC[0] & 0xff00;
ret |= wMAC[1] & 0x00ff;
return ret;
}
//------------------------------------------------------------------------------
//
// Function: x86KitlCreateName
//
// This function create device name from prefix and mac address (usually last
// two bytes of MAC address used for download).
//
BOOL x86KitlCreateName(CHAR *pPrefix, UINT16 mac[], CHAR *pBuffer)
{
int nLen;
DWORD dwUpperMAC = UpperDWFromMAC (mac);
// calculate total length needed
nLen = strlen (pPrefix) + 2 + 5; // 2 for vendor id, 5 for itoa of 16 bit value
if (OAL_KITL_ID_SIZE < nLen) {
NKDbgPrintfW (L"x86KitlCreateName: Device Name Too Long '%a' too long, can't create KITL name\r\n", pPrefix);
return FALSE;
}
strcpy (pBuffer, pPrefix);
strcat (pBuffer, FindNICAbbrev (dwUpperMAC));
itoa10 (((mac[2]>>8) | ((mac[2] & 0x00ff) << 8)), (pBuffer + strlen (pBuffer)), (OAL_KITL_ID_SIZE - strlen(pBuffer)));
RETAILMSG (1, (L"x86KitlCreateName: Using Device Name '%a'\r\n", pBuffer));
return TRUE;
}
extern void OEMGetRealTime (LPSYSTEMTIME pst);
static DWORD dwLastTime; // For making sure we aren't running backward
static DWORD dwBias;
extern BOOL g_fPostInit;
extern DWORD CurMSec;
DWORD
OEMEthGetSecs()
{
SYSTEMTIME st;
DWORD dwRet;
// after post-init, it's possible that someone is holding RTC CS, and we cannot safely call OEMGetRealTime while in KITL.
// Use CurMSec directly after post init.
// NOTE: if this function is called while interrupts are off, time not advance.
//
if (!g_fPostInit) {
OEMGetRealTime( &st );
dwRet = ((60UL * (60UL * (24UL * (31UL * st.wMonth + st.wDay) + st.wHour) + st.wMinute)) + st.wSecond);
dwBias = dwRet;
} else {
dwRet = (CurMSec/1000) + dwBias;
}
if (dwRet < dwLastTime) {
EdbgOutputDebugString("! Time went backwards (or wrapped): cur: %u, last %u\n",
dwRet,dwLastTime);
}
dwLastTime = dwRet;
return (dwRet);
}
UINT32 OALGetTickCount()
{
return OEMEthGetSecs () * 1000;
}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -