diff options
author | Jörg Frings-Fürst <debian@jff-webhosting.net> | 2014-07-06 18:04:32 +0200 |
---|---|---|
committer | Jörg Frings-Fürst <debian@jff-webhosting.net> | 2014-07-06 18:04:32 +0200 |
commit | a7f89980e5b3f4b9a74c70dbc5ffe8aabd28be28 (patch) | |
tree | 41c4deec1fdfbafd7821b4ca7a9772ac0abd92f5 /util/ipmidir.c |
Imported Upstream version 2.9.3upstream/2.9.3
Diffstat (limited to 'util/ipmidir.c')
-rw-r--r-- | util/ipmidir.c | 1513 |
1 files changed, 1513 insertions, 0 deletions
diff --git a/util/ipmidir.c b/util/ipmidir.c new file mode 100644 index 0000000..4865bc0 --- /dev/null +++ b/util/ipmidir.c @@ -0,0 +1,1513 @@ +/************************************************ + * + * ipmidir.c + * + * Supports direct raw KCS and SMBus user-space I/Os. + * Use this if no other IPMI driver is present. + * This interface would not be sufficient if more + * than one application is using IPMI at a time. + * This code is currently included for Linux, not for + * Windows. Windows requires imbdrv.sys or ipmidrv.sys. + * + * 08/21/06 Andy Cress - added as a new module + * 09/22/06 Andy Cress - improved SMBus base address detection + * 09/27/06 Andy Cress - fixed passing slave addrs other than BMC (0x20) + * 01/16/08 Andy Cress - more DBG messages + * + ************************************************/ +/*----------------------------------------------------------------------* +The BSD License + +Copyright (c) 2006, Intel Corporation +Copyright (c) 2009 Kontron America, Inc. +All rights reserved. + +Redistribution and use in source and binary forms, with or without +modification, are permitted provided that the following conditions are met: + + a.. Redistributions of source code must retain the above copyright notice, + this list of conditions and the following disclaimer. + b.. Redistributions in binary form must reproduce the above copyright notice, + this list of conditions and the following disclaimer in the documentation + and/or other materials provided with the distribution. + c.. Neither the name of Intel Corporation nor the names of its contributors + may be used to endorse or promote products derived from this software + without specific prior written permission. + +THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND +ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED +WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE +DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR +ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES +(INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; +LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON +ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS +SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + *----------------------------------------------------------------------*/ + +#if defined(STUB_IO) +/* May stub out direct io. For instance, PPC does not support <sys/io.h> */ +#define UCHAR unsigned char +#define UINT16 unsigned short +int ipmi_open_direct(int fdebugcmd) +{ return(-1); } +int ipmi_close_direct(void) +{ return(-1); } +int ipmi_cmdraw_direct(UCHAR cmd, UCHAR netfn, UCHAR lun, UCHAR sa, UCHAR bus, + UCHAR *pdata, int sdata, UCHAR *presp, + int *sresp, UCHAR *pcc, char fdebugcmd) +{ return(-1); } +int ipmi_cmd_direct(UINT16 icmd, UCHAR *pdata, int sdata, UCHAR *presp, + int *sresp, UCHAR *pcc, char fdebugcmd) +{ return(-1); } +int ipmi_set_max_kcs_loops(int ms) +{ return(0); } + +#elif defined(LINUX) || defined(BSD) || defined(DOS) || defined(MACOS) +#include <stdio.h> +#include <stdlib.h> +#include <fcntl.h> +#include <unistd.h> +#include <time.h> // for ProcessSendMessage timeout +#include <string.h> +#include <errno.h> +#ifdef DOS +#include <dos.h> +#else +#include <sys/mman.h> +#endif + +#include "imb_api.h" +#include "ipmicmd.h" +#include "ipmidir.h" +/* No special includes for 64-bit are needed. */ + +#if defined(LINUX) +#include <sys/io.h> +#elif defined(BSD) || defined(MACOS) +// #include <machine/cpufunc.h> +int iofd = -1; + +static __inline uchar +inbc(ushort port) +{ + uchar data; +/* from Linux sys/io.h: + * __asm__ __volatile__ ("inb %w1,%0":"=a" (data):"Nd" (port)); + * from BSD machine/cpufunc.h: + * __asm __volatile("inb %1,%0" : "=a" (data) : "id" ((ushort)(port))); + */ + __asm __volatile("inb %1,%0" : "=a" (data) : "d" (port)); + return (data); +} +static __inline void +outbc(ushort port, uchar data) +{ +/* from Linux sys/io.h: + * __asm__ __volatile__ ("outb %b0,%w1": :"a" (data), "Nd" (port)); + * from BSD machine/cpufunc.h: + * __asm __volatile("outb %0,%1" : : "a" (data), "id" ((ushort)(port))); + */ + __asm __volatile("outb %0,%1" : : "a" (data), "d" (port)); +} +static __inline uint +inl(ushort port) +{ + uint data; + + __asm __volatile("inl %%dx,%0" : "=a" (data) : "d" (port)); + return (data); +} +static __inline void +outl(ushort port, uint data) +{ + __asm __volatile("outl %0,%%dx" : : "a" (data), "d" (port)); +} +#endif +/* Linux & BSD define usleep() in unistd.h, but DOS has delay() instead. */ +#if defined(DOS) +void usleep(ulong usec) /*missing from DOS*/ +{ + delay(usec); +} +#endif + + +/* DBGP() is enabled with -x, DBGP2 is enabled if LINUX_DEBUG & -x */ +#define DBGP(fmt, args...) if (fdebugdir) fprintf(stdout,fmt, ##args) +#if defined(DOS) +#define DBGP2() ; /*no extra debug output*/ +#elif defined(LINUX_DEBUG) +#define DBGP2(fmt, args...) if (fdebugdir) fprintf(stderr,fmt, ##args) +#else +#define DBGP2(fmt, args...) ; /*no extra debug output*/ +#endif + +#ifdef ALONE +static int fjustpass = 0; +#else +// DRV_KCS, DRV_SMB types defined in ipmicmd.h +// extern int fDriverTyp; // use set_driver_type() instead +extern ipmi_cmd_t ipmi_cmds[NCMDS]; +extern int fjustpass; +#endif + +/*****************************************************************************/ +/* + * GLOBAL DATA + */ +/* KCS base addr: use 0xca2 for ia32, 0x8a2 for ia64 + * some use 0xca8/0xcac w register spacing + */ +#if defined(__ia64__) +/* gcc defines __ia64__, or Makefile.am defines __IA64__ */ +static UINT16 kcsBaseAddress = 0x8a2; /* for Itanium2 KCS */ +static UINT8 kcs_inc = 1; /*register spacing*/ +#else +static UINT16 kcsBaseAddress = 0xca2; /* for ia32 KCS */ +static UINT8 kcs_inc = 1; /*register spacing*/ +#endif +/* + * SMBus/SSIF: see dmidecode for I2C Slave Address & Base Addr. + * Slave Addr is constant for all Intel SSIF platforms (=0x84). + * 0x0540 base <= PCI ID 24D38086 at 00:1f.3 is JR (ICH5) + * 0x0400 base <= PCI ID 25A48086 at 00:1f.3 is TP (Hance_Rapids) + */ +UINT8 mBMCADDR = 0x84; /* SMBus I2C slave address = (0x42 << 1) */ +UINT16 mBMC_baseAddr = 0x540; /* usu. 0x0540 (JR), or 0x0400 if TP */ +UINT16 BMC_base = 0; /* resulting BMC base address (KCS or SMBus)*/ + +static char lock_dir_file[] = "/var/tmp/ipmiutil_dir_lock"; + +/* SSIF/SMBus defines */ +#define STATUS_REQ 0x60 +#define WR_START 0x61 +#define WR_END 0x62 +#define READ_BYTE 0x68 +#define COMMAND_REG (kcsBaseAddress+kcs_inc) +#define STATUS_REG (kcsBaseAddress+kcs_inc) +#define DATA_IN_REG (kcsBaseAddress) +#define DATA_OUT_REG (kcsBaseAddress) + +#if defined(DOS) +extern unsigned inp(unsigned _port); +extern unsigned outp(unsigned _port, unsigned _value); +#pragma intrinsic(inp, outp) +#define _INB(addr) inp((addr)) +#define _OUTB(data, addr) outp((addr),(data)) +#define _IOPL(data) 0 +#define ReadPortUchar( addr, valp ) (*valp) = inp((addr)) +#define WritePortUchar( addr, val ) outp((addr),(val)) +#define WritePortUlong( addr, val ) ( outp((addr),(val)) ) +#define ReadPortUlong( addr, valp ) (*(valp) = inp((addr)) ) +#elif defined(BSD) || defined(MACOS) +#define _INB(addr) inbc((addr)) +#define _OUTB(data, addr) outbc((addr),(data)) +#define _IOPL(data) 0 +#define ReadPortUchar( addr, valp ) (*valp) = inbc((addr)) +#define WritePortUchar( addr, val ) outbc((addr),(val)) +#define WritePortUlong( addr, val ) ( outl((addr),(val)) ) +#define ReadPortUlong( addr, valp ) (*(valp) = inl((addr)) ) +#else +#define _INB(addr) inb((addr)) +#define _OUTB(data, addr) outb((data),(addr)) +#define _IOPL(data) iopl((data)) +#define ReadPortUchar( addr, valp ) (*valp) = inb((addr)) +#define WritePortUchar( addr, val ) outb((val),(addr)) +#define WritePortUlong( addr, val ) ( outl((val), (addr)) ) +#define ReadPortUlong( addr, valp ) (*(valp) = inl((addr)) ) +#endif +/* Static data for Driver type, BMC type, etc. */ +static int g_DriverType = DRV_KCS; +static UINT16 g_ipmiVersion = IPMI_VERSION_UNKNOWN; +static UINT16 g_bmcType = BMC_UNKNOWN; +struct SMBCharStruct SMBChar; +static int fdebugdir = 0; +static char fDetectedIF = 0; + +/*****************************************************************************/ +/* Subroutine prototypes */ + +#ifdef ALONE +int set_driver_type(char * typ) { return; } +int get_IpmiStruct(UCHAR *iftype, UCHAR *ver, UCHAR *sa, int *base, UCHAR *inc) { return(-1); } +#else +/* extern int set_driver_type(char * typ); in ipmicmd.h */ +extern char fsm_debug; /*in mem_if.c*/ +extern int get_IpmiStruct(UCHAR *iftype, UCHAR *ver, UCHAR *sa, int *base, UCHAR *inc); +#endif +int SendTimedImbpRequest_kcs( IMBPREQUESTDATA *requestData, + unsigned int timeout, UINT8 *resp_data, + int *respDataLen, UINT8 *compCode); +int SendTimedImbpRequest_ssif( IMBPREQUESTDATA *requestData, + unsigned int timeout, UINT8 *resp_data, + int *respDataLen, UINT8 *compCode); +int ImbInit_dir(void); +static int SendmBmcRequest( + unsigned char* request, UINT32 reqLength, + unsigned char* response, UINT32* respLength, + UINT32 ipmiOldFlag); +static int ReadResponse( unsigned char* resp, struct SMBCharStruct* smbChar); +static int SendRequest( unsigned char* req, int length, + struct SMBCharStruct* smbChar ); +static int GetDeviceId(UINT16 *bmcType, UINT16 *ipmiVersion); +static int send_raw_kcs(UINT8 *rqData, int rqLen, UINT8 *rsData, int *rsLen ); +static int ProcessTimedMessage(BMC_MESSAGE *p_reqMsg, BMC_MESSAGE *p_respMsg, const UINT32 timeout); +static int wait_for_SMS_flag(void); + +/*****************************************************************************/ +/* Subroutines */ + +static int ProcessMessage(BMC_MESSAGE *p_reqMsg, BMC_MESSAGE *p_respMsg) +{ + int status = STATUS_OK; + UINT32 timeout = 1000 * GETMSGTIMEOUT; // value is in milliseconds + status = ProcessTimedMessage(p_reqMsg, p_respMsg, timeout); + return status; +} + +static char *BmcDesc(unsigned char drvtype) +{ /* return BMC Driver type description string */ + char *msg; + switch(drvtype) { + case DRV_KCS: msg = "KCS"; break; /*BMC Sahalee KCS*/ + case DRV_SMB: msg = "SMBus"; break; /*mBMC SMBus (SSIF)*/ + default: msg = ""; + } + return(msg); +} + +int get_ipmi_if(void) +{ + /* Only care about Linux since ipmidir is not used for Windows. */ + char *if_file = "/var/lib/ipmiutil/ipmi_if.txt"; + char *if_file2 = "/usr/share/ipmiutil/ipmi_if.txt"; + FILE *fp; + char line[80]; + char *p; + char *r; + int rv = ERR_NO_DRV; + int i, j; + ulong mybase = 0; + int myinc = 1; + // uchar *rgbase; + + fp = fopen(if_file,"r"); + if (fp == NULL) fp = fopen(if_file2,"r"); + if (fp == NULL) { /*error opening ipmi_if file*/ + return -1; + } else { /*read the data from the ipmi_if file*/ + while ( (p = fgets(line,sizeof(line),fp)) != NULL) + { + if (strstr(line,"Interface type:") != NULL) { + if (strstr(line,"KCS") != NULL) { + g_DriverType = DRV_KCS; + } else { + g_DriverType = DRV_SMB; + } + } else if (strstr(line,"Base Address:") != NULL) { + r = strchr(line,':'); + i = strspn(++r," \t"); /* skip leading spaces */ + r += i; + j = strcspn(r," \t\r\n"); /* end at next whitespace */ + r[j] = 0; /*stringify*/ + /* convert the "0x0000000000000401" string to an int */ + if (strncmp(r,"0x",2) == 0) { r += 2; j -= 2; } + DBGP2("base addr 0x: %s, mybase=%x j=%d\n",r,mybase,j); + mybase = strtol(r, NULL, 16); /*hex string is base 16*/ + DBGP2("base addr 0x: %s, mybase=%x \n",r,mybase); + } else if (strstr(line,"Register Spacing:") != NULL) { + r = strchr(line,':'); + i = strspn(++r," \t"); /* skip leading spaces */ + r += i; + j = strcspn(r," \t"); /* end at next whitespace */ + r[j] = 0; /*stringify*/ + myinc = atoi(r); + } + } + fclose(fp); + DBGP("ipmi_if: Driver = %d (%s), Base = 0x%04lx, Spacing = %d\n", + g_DriverType,BmcDesc(g_DriverType),mybase,myinc); + if (g_DriverType == DRV_SMB) { + if (mybase & 0x0001) mybase -= 1; /* 0x0401 -> 0x0400 */ + if (mybase != 0 && (mybase & 0x0F) == 0) { /* valid base address */ + mBMC_baseAddr = mybase; + BMC_base = mBMC_baseAddr; + rv = 0; + } + } else { /*KCS*/ + if (mybase != 0) { /* valid base address */ + kcsBaseAddress = (mybase & 0x0000ffff); + BMC_base = kcsBaseAddress; + if (myinc > 1) kcs_inc = myinc; + rv = 0; + } + } + DBGP2("ipmi_if: BMC_base = 0x%04x, kcs_inc = %d, ret = %d\n", + BMC_base, kcs_inc,rv); + } + return(rv); +} + +static int set_lock_dir(void) +{ + int rv = 0; + FILE *fp; + fp = fopen(lock_dir_file,"w"); + if (fp == NULL) rv = -1; + else { + fclose(fp); + rv = 0; + } + return(rv); +} + +static int clear_lock_dir(void) +{ + int rv = 0; + rv = remove(lock_dir_file); /*same as unlink() */ + if (fdebugdir) printf("clear_lock rv = %d\n",rv); + return(rv); +} + +static int check_lock_dir(void) +{ + int rv = 0; +#ifdef LOCK_OK + FILE *fp; + fp = fopen(lock_dir_file,"r"); + if (fp == NULL) rv = 0; + else { + fclose(fp); + rv = -1; /*error, lock file exists*/ + } +#endif + return(rv); +} + +int ipmi_open_direct(int fdebugcmd) +{ + int status = 0; + //char *dmsg = ""; + int i; + + DBGP2("open_direct(%d) called\n",fdebugcmd); + if (fdebugcmd) { + fdebugdir = fdebugcmd; +#if defined(LINUX_DEBUG) && !defined(ALONE) + fsm_debug = fdebugcmd; +#endif + } + + /* Read ipmi_if config file data, if present. */ + status = get_ipmi_if(); + if (status == -1) { + uchar iftype, iver, sa, inc; + int mybase; + /* Read SMBIOS to get IPMI struct */ + status = get_IpmiStruct(&iftype,&iver,&sa,&mybase,&inc); + if (status == 0) { + if (iftype == 0x04) { + g_DriverType = DRV_SMB; + mBMC_baseAddr = mybase; + } else { /*0x01==KCS*/ + g_DriverType = DRV_KCS; + if (sa == 0x20 && mybase != 0) { /*valid*/ + kcsBaseAddress = mybase; + kcs_inc = inc; + } + } + BMC_base = mybase; + DBGP("smbios: Driver=%d(%s), sa=%02x, Base=0x%04x, Spacing=%d\n", + g_DriverType,BmcDesc(g_DriverType),sa,mybase,inc); + } + } + +#ifndef DOS + /* superuser/root priv is required for direct I/Os */ + i = geteuid(); /*direct is Linux only anyway*/ + if (i > 1) { + fprintf(stdout,"Not superuser (%d)\n", i); + return ERR_NO_DRV; + } +#endif + /* check lock for driverless interface */ + i = check_lock_dir(); + if (i != 0) { + fprintf(stdout,"open_direct interface locked, %s in use\n", + lock_dir_file); + return ERR_NO_DRV; + } + + /* Find the SMBIOS IPMI driver type, data */ + status = ImbInit_dir(); + DBGP2("open_direct Init status = %d\n",status); + DBGP2("open_direct base=%x spacing=%d\n",BMC_base,kcs_inc); + if (status == 0) { + fDetectedIF = 1; /*Successfully detected interface */ + /* Send a command to the IPMI interface */ + if (!fjustpass) + status = GetDeviceId(&g_bmcType,&g_ipmiVersion); + if (status == 0) { + char *typ; + if (g_DriverType == DRV_SMB) typ = "smb"; + else typ = "kcs"; + set_driver_type(typ); + } + } + DBGP("open_direct: status=%d, %s drv, ipmi=%d\n", + status,BmcDesc(g_DriverType),g_ipmiVersion); + + /* set lock for driverless interface */ + i = set_lock_dir(); + return status; +} + +int ipmi_close_direct(void) +{ + int status = 0; +#if defined(BSD) || defined(MACOS) + close(iofd); + iofd = -1; +#endif + /* clear lock for driverless interface */ + status = clear_lock_dir(); + return status; +} + +int ipmi_cmdraw_direct(UCHAR cmd, UCHAR netfn, UCHAR lun, UCHAR sa, UCHAR bus, + UCHAR *pdata, int sdata, UCHAR *presp, + int *sresp, UCHAR *pcc, char fdebugcmd) +{ + BMC_MESSAGE sendMsg; + BMC_MESSAGE respMsg; + int status; + int len = 0; + + if (g_bmcType == BMC_UNKNOWN) { + /* User-specified a driver type, but need open */ + status = ipmi_open_direct(fdebugcmd); + } + fdebugdir = fdebugcmd; + if (sdata > IPMI_REQBUF_SIZE) return(LAN_ERR_BADLENGTH); + if (fjustpass) { + status = send_raw_kcs(pdata, sdata, presp, sresp); + *pcc = 0; + return(status); + } + sendMsg.Bus = bus; + sendMsg.DevAdd = sa; + sendMsg.NetFn = netfn; + sendMsg.LUN = lun; + sendMsg.Cmd = cmd; + sendMsg.Len = sdata; + if (sdata > 0) memcpy(sendMsg.Data,pdata,sdata); + + status = ProcessMessage(&sendMsg, &respMsg); + if (status == STATUS_OK) { + *pcc = respMsg.CompCode; + if (respMsg.Len > 0) { + len = respMsg.Len; + if (len > *sresp) len = *sresp; + } else len = 0; + if (len > 0) + memcpy(presp,respMsg.Data,len); + *sresp = len; + } + return(status); +} + +#ifndef ALONE +int ipmi_cmd_direct(UINT16 icmd, UCHAR *pdata, int sdata, UCHAR *presp, + int *sresp, UCHAR *pcc, char fdebugcmd) +{ + UINT8 cmd, netfn, sa, lun, bus; + int status; + int i; + + fdebugdir = fdebugcmd; + for (i = 0; i < NCMDS; i++) { + if (ipmi_cmds[i].cmdtyp == icmd) { + cmd = (icmd & 0x00ff); + sa = ipmi_cmds[i].sa; + bus = ipmi_cmds[i].bus; + netfn = ipmi_cmds[i].netfn; + lun = ipmi_cmds[i].lun; + break; + } + } + if (i >= NCMDS) { + DBGP("ipmidir: icmd %04x not found, defaults used\n",icmd); + cmd = (icmd & 0xFF); + netfn = (icmd & 0xFF00) >> 8; + sa = BMC_ADDR; + lun = BMC_LUN; + bus = 0; + } + status = ipmi_cmdraw_direct(cmd, netfn, lun, sa, bus, pdata, sdata, + presp, sresp, pcc, fdebugcmd); + return(status); +} +#endif + +static UINT8 +CalculateChecksum(UINT8* p_buff, int length) +{ + UINT8 sum = 0; + int i; + for (i = 0; i < length; i++) sum+= p_buff[i]; + return (~sum) + 1; +} + +static int +GetDeviceId(UINT16 *bmcType, UINT16 *ipmiVersion) +{ + int status = STATUS_OK; + UINT16 thisBmcType = BMC_UNKNOWN; + BMC_MESSAGE sendMsg; + BMC_MESSAGE respMsg; + + DBGP2("open_direct: detecting interface, bmc: %X ver: %X drv=%s\n", + g_bmcType, g_ipmiVersion, BmcDesc(g_DriverType)); + + // if the BMC type has not yet been detected - detect it + if( g_bmcType == BMC_UNKNOWN ) + { + // Try sending the GetDeviceId command to the default interface + sendMsg.DevAdd = BMC_ADDR; + sendMsg.NetFn = NETFN_APP; + sendMsg.LUN = BMC_LUN; + sendMsg.Cmd = CMD_GET_DEVICE_ID; + sendMsg.Len = 0; + DBGP2("open_direct: Try Get_Device_ID with %s driver\n", + BmcDesc(g_DriverType)); + status = ProcessMessage(&sendMsg, &respMsg); + if (status == STATUS_OK) { + if (g_DriverType == DRV_KCS) thisBmcType = BMC_SAHALEE; + else thisBmcType = BMC_MBMC_87431M; + } else { /*error, switch interfaces*/ + DBGP("open_direct: ProcessMessage(%s) error = %d\n", + BmcDesc(g_DriverType),status); + if (!fDetectedIF) { + /* if not yet detected, try other IF type */ + if (g_DriverType == DRV_KCS) { + DBGP2("open_direct: Not KCS, try SSIF/SMBus\n"); + g_DriverType = DRV_SMB; + } else { + DBGP2("open_direct: Not SSIF, try KCS\n"); + g_DriverType = DRV_KCS; + } + } + } /*end-else error*/ + + // If error, try again with the other interface + if (thisBmcType == BMC_UNKNOWN) + { + // send the command via KCS/SMBus to get the IPMI version. + status = ProcessMessage(&sendMsg, &respMsg); + if (status == STATUS_OK) { + if (g_DriverType == DRV_KCS) thisBmcType = BMC_SAHALEE; + else thisBmcType = BMC_MBMC_87431M; + } else { + // If sending message fails + status = ER_NO_BMC_IF; + } + } + g_bmcType = thisBmcType; + + // if we contacted the BMC, decode its version + if( status == STATUS_OK ) + { + if( g_bmcType == BMC_MBMC_87431M ) + { + // Decode the miniBMC productID + } + if( respMsg.Data[4] == 0x51 ) + g_ipmiVersion = (UINT16)IPMI_VERSION_1_5; + else if( respMsg.Data[4] == 0x02 ) + g_ipmiVersion = IPMI_VERSION_2_0; + } + } /* endif unknown BMC */ + + // set the version info and leave + *ipmiVersion = g_ipmiVersion; + *bmcType = g_bmcType; + + DBGP2("open_direct: AFTER bmc: %X ver: %X\n", + g_bmcType, g_ipmiVersion); + return status; +} + +//***************************************************************************** +#if defined (WIN64) + // Do Nothing for WIN64 as we are not using these functions here +#else +static int ProcessSendMessage(BMC_MESSAGE *p_reqMsg, BMC_MESSAGE *p_respMsg, const UINT8 bus, const UINT8 slave, const UINT32 timeout) +{ + int status = STATUS_OK; + BMC_MESSAGE sendReq; + static UINT8 sendSeq = 1; + static UINT8 incTestCount=0; + BMC_MESSAGE reqMsg,respMsg; + int retryCount; + UINT32 i; + UINT nRetryCount = 0; + int testCount; + + // Windows will be using Async Imb request interface to poll for + // messages in the BMC SMS message queue + + // format the send message packet + sendReq.Cmd = SENDMESSAGE_CMD; + sendReq.DevAdd = BMC_ADDR; + sendReq.LUN = 0; + sendReq.NetFn = NETFN_APP; + + sendReq.Data[0] = bus; + sendReq.Data[1] = slave; + // NetFn is the upper 6 bits of the data byte, the LUN is the lower two + sendReq.Data[2] = ((p_reqMsg->NetFn << 2) | (p_reqMsg->LUN & 0x03)); + sendReq.Data[3] = CalculateChecksum(&sendReq.Data[1], 2); + sendReq.Data[4] = BMC_ADDR; + // NetFn is the upper 6 bits of the data byte, the LUN is the lower two + sendReq.Data[5] = ((sendSeq << 2) | (SMS_MSG_LUN & 0x03)); + // sequence number = 1 << 2 and BMC message response lun = 0x02 + sendReq.Data[6] = p_reqMsg->Cmd; + + // loop to copy the command data to the send message data format + i = 0; + for(i=0; i < p_reqMsg->Len; i++) + sendReq.Data[7+i] = p_reqMsg->Data[i]; + + // create a checksum + sendReq.Data[7+i] = CalculateChecksum(&sendReq.Data[4], p_reqMsg->Len + 3); // send data length plus the values from index 3 - 6 + + // send length plus 0 - 7 and checksum byte + sendReq.Len = p_reqMsg->Len + 8; + /* + * Send the message with retries + * For get Device ID & Read FRU data, retry less, in case + * HSC/LCP is not present, + */ + if (p_reqMsg->NetFn != 0x08 ) + retryCount = BMC_MAX_RETRIES+2; + else + retryCount = BMC_MAX_RETRIES+20; + do + { // send the message + if( (status = ProcessMessage(&sendReq, p_respMsg)) == STATUS_OK ) + { + // some error, maybe because the controller was not ready yet. + if( p_respMsg->CompCode == 0x83 ) { + // Sleep for 1 second and try again + sleep(1); + incTestCount = 1; // true + continue; // try again + } + else if( p_respMsg->CompCode == 0x82 ) { + DBGP("ProcessSendMessage(sa=%02x,%02x,%02x) " + "ccode=82 bus error\n", + slave,p_reqMsg->NetFn,p_reqMsg->Cmd); + status = ERGETTINGIPMIMESSAGE; + break; // exit with error + } + else if( p_respMsg->CompCode != 0x00 ) + continue; //break; // exit with error + + status = wait_for_SMS_flag(); /* new, added */ + if (status == -1) + DBGP("wait_for_SMS_flag timeout\n"); + + // Only for windows we use Imb Asyn interface for polling + // For Unix we issue GetMessage command + // issue a getmessage command + reqMsg.DevAdd = BMC_ADDR; + reqMsg.NetFn = NETFN_APP; + reqMsg.LUN = BMC_LUN; + reqMsg.Cmd = GETMESSAGE_CMD; + reqMsg.Len = 0; + + nRetryCount = 0; + testCount = 100; /* For HSC */ + if (slave == 0x22 || incTestCount == 1) // true + testCount=1000; /*added for LCP */ + do + { + /* Loop here for the response with the correct + * sequence number. */ + if( (status = ProcessMessage( &reqMsg, &respMsg )) != 0 ) { + DBGP("Breaking after Getmsg, Status is %d\n",status); + break; + } + if( slave == 0x22 && p_reqMsg->Cmd == 0x04 ) { + /* give debug if LCP ever gets here. */ + DBGP("LCP get: cnt[%d,%d] seq[%02x,%02x] cc=%02x status=%u\n", + testCount,retryCount, sendSeq, + ((respMsg.Data[4] & 0xFC)>>2), + respMsg.CompCode,status) ; + return STATUS_OK; + } + if (respMsg.CompCode != 0) { + DBGP2("get, cnt[test=%d] slave=%02x, cc==%02x\n", + testCount,slave,respMsg.CompCode); + /* Used to wait 1 ms via usleep(1000), but not needed. */ + // usleep(1000); + } + } while(((respMsg.CompCode == 0x83 ||(respMsg.CompCode == 0x80)) + && --testCount > (int) 0) || (respMsg.CompCode == 0x0 + && ((respMsg.Data[4] & 0xFC) >> 2) != sendSeq) ); + + DBGP("get cnt[test=%d,retry=%d] seq[Send=0x%02x Recv=0x%02x] " + "CompCode=0x%x status=%u\n", testCount,retryCount, + sendSeq,((respMsg.Data[4] & 0xFC)>>2), + respMsg.CompCode,status); + + if( (respMsg.CompCode == 0x00) && (status == STATUS_OK) && + ((respMsg.Data[4] & 0xFC) >> 2 == sendSeq) ) + { + // format the GetMessage response + (*p_respMsg) = (*p_reqMsg); + // The first data byte is the channel number the message + // was sent on + p_respMsg->DevAdd = respMsg.Data[3]; + p_respMsg->NetFn = respMsg.Data[1] >> 2; + p_respMsg->LUN = respMsg.Data[4] & 0x03; + p_respMsg->Cmd = respMsg.Data[5]; + p_respMsg->CompCode = respMsg.Data[6]; // comp code + p_respMsg->Len = respMsg.Len - 8; + // the last data byte is the checksum + if ( p_respMsg->CompCode == 0xCC) { + DBGP2(" Cmd=%d Len=%d ccode=CC ", + p_respMsg->Cmd, p_respMsg->Len); + } + for(i=0; i < p_respMsg->Len; i++) + p_respMsg->Data[i] = respMsg.Data[7+i]; + break; + } + } + // this will retry the read the number of times in retryCount + } while( --retryCount > 0 ); + + if( retryCount <= 0 ) status = ERGETTINGIPMIMESSAGE; + + /* Increment the sequence number + * sequence number can't be greater 63 (6bit number), + * so wrap the counter if it is */ + ++sendSeq; + if( sendSeq >= 64 ) sendSeq = 1; + return status; +} + +#endif /* All other than WIN64 */ + +/* #define MAX_KCS_LOOP 30000 *was 7000*/ +/* Set a failsafe MAX KCS Loops to prevent infinite loop on status reg */ +/* max was 7000, but need longer for ClearSEL cmd */ +static int max_kcs_loop = 30000; /*means 300ms*/ +static int max_sms_loop = 500; /*means 500ms*/ +static int peak_loops = 0; +int ipmi_set_max_kcs_loops(int ms) +{ + max_kcs_loop = ms * 100; /*300 ms * 100 = 30000 loops*/ + max_sms_loop = ms; + return 0; +} + +static int wait_for_SMS_flag(void) +{ + int i = 0; + while((_INB(STATUS_REG) & 0x04) == 0) + { + usleep(2 * 1000); /*sleep for 2 msec*/ + i += 2; + if ( i > max_sms_loop ) return -1; + } + return 0; +} + +static int wait_for_IBF_clear(void) + { + int i = 0; + while ((_INB(STATUS_REG) & 0x02) == 0x02) { + if (i > 0 && (i % 100) == 0) usleep(1000); /*sleep for 1 msec*/ + if (i > max_kcs_loop) { + DBGP("wait_for_IBF_clear: max loop %d\n",i); + return -1; + // break; + } + i++; + } + if (i > peak_loops) peak_loops = i; + return 0; + } + +static int wait_for_OBF_set(void) + { + int i = 0; + while ((_INB(STATUS_REG) & 0x01) == 0x00) { + if (i > 0 && (i % 100) == 0) usleep(1000); /*sleep for 1 msec*/ + if (i > max_kcs_loop) { + DBGP("wait_for_OBF_set: max loop %d\n",i); + return -1; + // break; + } + i++; + } + if (i > peak_loops) peak_loops = i; + return 0; + } + +static inline int get_write_state(void) + { + if ((_INB(STATUS_REG) >> 6) != 0x02) + return -1; + return 0; + } + +static inline int get_read_state(void) + { + if ((_INB(STATUS_REG) >> 6) != 0x01) + return -1; + return 0; + } + +static inline int get_idle_state(void) + { + if ((_INB(STATUS_REG) >> 6) != 0x00) + return -1; + return 0; + } + +UINT8 dummy2; +static inline void clear_OBF(void) +{ + dummy2 = _INB(DATA_IN_REG); +} + +static int +send_raw_kcs (UINT8 *rqData, int rqLen, UINT8 *rsData, int *rsLen ) +{ + int length; + unsigned char dummy; + unsigned char rx_data[64]; + int rxbuf_len; + int rv; + + if (fdebugdir) { + int cnt; + DBGP("send_raw_kcs: "); + for ( cnt=0; cnt < rqLen ; cnt++) + DBGP(" %02x",rqData[cnt]); + DBGP("\n"); + } + wait_for_IBF_clear(); + clear_OBF(); + _OUTB(WR_START,COMMAND_REG); + rv = wait_for_IBF_clear(); + if (get_write_state() != 0) return LAN_ERR_SEND_FAIL; + clear_OBF(); + if (rv != 0) return LAN_ERR_SEND_FAIL; + + for(length = 0;length < rqLen-1;length++) + { + _OUTB(rqData[length],DATA_OUT_REG); + wait_for_IBF_clear(); + if (get_write_state() != 0) + return LAN_ERR_SEND_FAIL; + clear_OBF(); + } + + _OUTB(WR_END,COMMAND_REG); + wait_for_IBF_clear(); + if (get_write_state() != 0) + return LAN_ERR_SEND_FAIL; + clear_OBF(); + _OUTB(rqData[length],DATA_OUT_REG); + + /* write phase complete, start read phase */ + rxbuf_len = *rsLen; + *rsLen = 0; + while(*rsLen <= IPMI_RSPBUF_SIZE) + { + wait_for_IBF_clear(); + if (get_read_state() != 0) + { + if (get_idle_state() != 0) { + DBGP2("not idle in rx_data (%02x)\n",_INB(STATUS_REG)); + // clear_lock_dir(); + return LAN_ERR_RECV_FAIL; + } else { + rv = wait_for_OBF_set(); + if (rv != 0) return LAN_ERR_RECV_FAIL; + dummy = _INB(DATA_IN_REG); + /* done, copy the data */ + for (length=0;length < *rsLen;length++) + rsData[length] = rx_data[length]; + return ACCESS_OK; + } + } else { + rv = wait_for_OBF_set(); + if (rv != 0) return LAN_ERR_RECV_FAIL; + rx_data[*rsLen] = _INB(DATA_IN_REG); + DBGP2("rx_data[%d] is 0x%x\n",*rsLen,rx_data[*rsLen]); + _OUTB(READ_BYTE,DATA_IN_REG); + (*rsLen)++; + } + if (*rsLen > rxbuf_len) { + DBGP("ipmidir: rx buffer overrun, size = %d\n",rxbuf_len); + break; /*stop if user buffer max*/ + } + } /*end while*/ + return ACCESS_OK; +} + +/* + * SendTimedImbpRequest_kcs - write bytes to KCS interface, read response + * The bytes are written in this order: + * 1 netfn + * 2 cmdType + * 3-N data, if any + */ +int +SendTimedImbpRequest_kcs (IMBPREQUESTDATA *requestData, + unsigned int timeout, UINT8 *resp_data, int *respDataLen, + unsigned char *compCode) +{ /*SendTimedImb for KCS*/ + int length; + unsigned char dummy; + unsigned char rx_data[64]; + int rxbuf_len, rv; + + if (fdebugdir) { + int cnt; + DBGP("Send Netfn=%02x Cmd=%02x, raw: %02x %02x %02x %02x", + requestData->netFn, requestData->cmdType, + requestData->busType, requestData->rsSa, + (requestData->netFn<<2), requestData->cmdType); + for ( cnt=0; cnt < requestData->dataLength ; cnt++) + DBGP(" %02x",requestData->data[cnt]); + DBGP("\n"); + } + + rv = wait_for_IBF_clear(); + clear_OBF(); + if (rv != 0) return LAN_ERR_SEND_FAIL; + _OUTB(WR_START,COMMAND_REG); + rv = wait_for_IBF_clear(); + if (get_write_state() != 0) return LAN_ERR_SEND_FAIL; + clear_OBF(); + if (rv != 0) return LAN_ERR_SEND_FAIL; + + _OUTB((requestData->netFn << 2),DATA_OUT_REG); + rv = wait_for_IBF_clear(); + if (get_write_state() != 0) return LAN_ERR_SEND_FAIL; + clear_OBF(); + + if (requestData->dataLength == 0) + { + _OUTB(WR_END,COMMAND_REG); + wait_for_IBF_clear(); + if (get_write_state() != 0) + return LAN_ERR_SEND_FAIL; + clear_OBF(); + + _OUTB(requestData->cmdType,DATA_OUT_REG); + } + else + { + + _OUTB(requestData->cmdType,DATA_OUT_REG); + wait_for_IBF_clear(); + if (get_write_state() != 0) + return LAN_ERR_SEND_FAIL; + clear_OBF(); + + for(length = 0;length < requestData->dataLength-1;length++) + { + _OUTB(requestData->data[length],DATA_OUT_REG); + wait_for_IBF_clear(); + if (get_write_state() != 0) + return LAN_ERR_SEND_FAIL; + clear_OBF(); + } + + _OUTB(WR_END,COMMAND_REG); + wait_for_IBF_clear(); + if (get_write_state() != 0) + return LAN_ERR_SEND_FAIL; + clear_OBF(); + + _OUTB(requestData->data[length],DATA_OUT_REG); + } + +/********************************** WRITE PHASE OVER ***********************/ + +#ifdef TEST_ERROR + if (fdebugdir == 5) { /*introduce an error test case*/ + printf("Aborting after KCS write, before read\n"); + return(rv); + } +#endif +// length = 0; +// usleep(100000); +/********************************** READ PHASE START ***********************/ + rxbuf_len = *respDataLen; + *respDataLen = 0; + + while(*respDataLen <= IPMI_RSPBUF_SIZE) + { + wait_for_IBF_clear(); + if (get_read_state() != 0) + { + if (get_idle_state() != 0) { + DBGP2("not idle in rx_data (%02x)\n",_INB(STATUS_REG)); + // clear_lock_dir(); + return LAN_ERR_RECV_FAIL; + } else { + rv = wait_for_OBF_set(); + if (rv != 0) { return LAN_ERR_RECV_FAIL; } + dummy = _INB(DATA_IN_REG); + /* done, copy the data, if valid */ + if (*respDataLen < 3) { /* data not valid, no cc */ + (*respDataLen) = 0; + *compCode = 0xCA; /*cannot return #bytes*/ + // clear_lock_dir(); + return LAN_ERR_TIMEOUT; + } else { /*valid*/ + requestData->netFn = rx_data[0]; + requestData->cmdType = rx_data[1]; + *compCode = rx_data[2]; + (*respDataLen) -= 3; + for (length=0;length < *respDataLen;length++) + resp_data[length] = rx_data[length+3]; + } + DBGP2("ipmidir: peak_loops = %d\n",peak_loops); + return ACCESS_OK; + } + } else { + rv = wait_for_OBF_set(); + if (rv != 0) { return LAN_ERR_RECV_FAIL; } + rx_data[*respDataLen] = _INB(DATA_IN_REG); + DBGP2("rx_data[%d] is 0x%x\n",*respDataLen,rx_data[*respDataLen]); + _OUTB(READ_BYTE,DATA_IN_REG); + (*respDataLen)++; + } + if (*respDataLen > rxbuf_len) { + DBGP("ipmidir: rx buffer overrun, size = %d\n",rxbuf_len); + break; /*stop if user buffer max*/ + } + } /*end while*/ + return ACCESS_OK; +} /* end SendTimedImbpRequest_kcs */ + + +static int ProcessTimedMessage(BMC_MESSAGE *p_reqMsg, BMC_MESSAGE *p_respMsg, const UINT32 timeout) +{ + int status = STATUS_OK; + IMBPREQUESTDATA requestData = {0}; + int respDataLen = IPMI_RSPBUF_SIZE; + UINT8 compCode = 0; + // static UINT8 sendSeq = 1; + // UINT8 buff[DATA_BUF_SIZE] = {0}; + ACCESN_STATUS accessn; + int i, j; + + j = p_reqMsg->Len; + if (j > IPMI_REQBUF_SIZE) return(LAN_ERR_BADLENGTH); + // Initialize Response Message Data + for(i = 0; i < IPMI_RSPBUF_SIZE; i++) + { + p_respMsg->Data[i] = 0; + } + + /* show the request */ + DBGP("ipmidir Cmd=%02x NetFn=%02x Lun=%02x Sa=%02x Data(%d): ", + p_reqMsg->Cmd, p_reqMsg->NetFn, + p_reqMsg->LUN, p_reqMsg->DevAdd, j); + for (i=0; i<j; i++) DBGP("%02x ",p_reqMsg->Data[i]); + DBGP("\n"); + + j = _IOPL(3); + if (j != 0) { + DBGP("ipmi_direct: iopl errno = %d\n",errno); + return(errno); + } + // Initializes Request Message + + // Call into IPMI + if (p_reqMsg->DevAdd == 0x20) { + requestData.cmdType = p_reqMsg->Cmd; + requestData.rsSa = 0x20; + requestData.busType = 0; + requestData.netFn = p_reqMsg->NetFn; + requestData.rsLun = p_reqMsg->LUN; + requestData.data = p_reqMsg->Data; + requestData.dataLength = (int)p_reqMsg->Len; + + if (g_DriverType == DRV_KCS) /*KCS*/ + accessn = SendTimedImbpRequest_kcs(&requestData, timeout, + p_respMsg->Data, &respDataLen, &compCode); + else if (g_DriverType == DRV_SMB) /*SMBus*/ + accessn = SendTimedImbpRequest_ssif(&requestData, timeout, + p_respMsg->Data, &respDataLen, &compCode); + else { /* should never happen */ + printf("ipmi_direct: g_DriverType invalid [%d]\n",g_DriverType); + return(ERR_NO_DRV); + } + + status = accessn; + // Return Response Message + p_respMsg->DevAdd = p_reqMsg->DevAdd; + p_respMsg->NetFn = requestData.netFn; + p_respMsg->LUN = p_reqMsg->LUN; + p_respMsg->Cmd = requestData.cmdType; + p_respMsg->CompCode = compCode; + p_respMsg->Len = respDataLen; + } else { /*DevAdd != 0x20*/ + status = ProcessSendMessage(p_reqMsg, p_respMsg, p_reqMsg->Bus, + p_reqMsg->DevAdd,10000); + DBGP2("ProcessSendMessage(cmd=%02x,rs,sa=%02x,10000) = %d\n", + p_reqMsg->Cmd,p_reqMsg->DevAdd,status); + } + + /* validate the response data length */ + if (p_respMsg->Len > IPMI_RSPBUF_SIZE) p_respMsg->Len = IPMI_RSPBUF_SIZE; + /* show the response */ + j = p_respMsg->Len; + DBGP("ipmidir Resp(%x,%x): status=%d cc=%02x, Data(%d): ", + (p_respMsg->NetFn >> 2), p_respMsg->Cmd, + status,p_respMsg->CompCode, j); + if (status == 0) + for (i=0; i<j; i++) DBGP("%02x ",p_respMsg->Data[i]); + DBGP("\n"); + + return status; +} + +/* + * ImbInit_dir + * Uses SMBIOS to determine the driver type and base address. + * It also checks the status register if KCS. + */ +int ImbInit_dir(void) +{ + uchar v = 0xff; + + /* Read SMBIOS to get IPMI struct */ + DBGP2("ImbInit: BMC_base = 0x%04x\n",BMC_base); + if (BMC_base == 0) { /*use get_IpmiStruct routine from mem_if.c */ + uchar iftype, iver, sa, inc; + int mybase, status; + char *ifstr; + status = get_IpmiStruct(&iftype,&iver,&sa,&mybase,&inc); + if (status == 0) { + if (iftype == 0x04) { + g_DriverType = DRV_SMB; + ifstr = "SSIF"; + mBMC_baseAddr = mybase; + } else /*0x01==KCS*/ { + g_DriverType = DRV_KCS; + ifstr = "KCS"; + if (sa == BMC_SA && mybase != 0) { /*valid*/ + kcsBaseAddress = mybase; + kcs_inc = inc; + } + } + BMC_base = mybase; + DBGP("SMBIOS IPMI Record found: type=%s sa=%02x base=0x%04x spacing=%d\n", + ifstr, sa, mybase, inc); + } + } + + /* Use KCS here. There are no known SMBus implementations on 64-bit */ + if (BMC_base == 0) { + DBGP("No IPMI Data Structure Found in SMBIOS Table,\n"); + g_DriverType = DRV_KCS; + BMC_base = kcsBaseAddress; + DBGP("Continuing with KCS on Default Port 0x%04x\n",kcsBaseAddress); + } +#if defined(BSD) || defined(MACOS) + iofd = open("/dev/io",O_RDWR); + if (iofd < 0) { + printf("Cannot open /dev/io...Exiting\n"); + return ERR_NO_DRV; + } +#endif + if (g_DriverType == DRV_SMB) { + /* Perhaps add controller type in ipmi_if.txt (?)*/ + /* Intel SSIF: 0x0540=SJR, 0x0400=STP */ + if (mBMC_baseAddr == 0x540 || mBMC_baseAddr == 0x400) + SMBChar.Controller = INTEL_SMBC; + else /*else try ServerWorks*/ + SMBChar.Controller = SW_SMBC; + SMBChar.baseAddr = mBMC_baseAddr; + DBGP("BMC SSIF/SMBus Interface at i2c=%02x base=0x%04x\n", + mBMCADDR,mBMC_baseAddr); + } + if (g_DriverType == DRV_KCS) { + v = _IOPL(3); + v = _INB(STATUS_REG); + DBGP2("inb(%x) returned %02x\n",STATUS_REG,v); + if (v == 0xff) { + printf("No Response from BMC...Exiting\n"); + return ERR_NO_DRV; + } + DBGP("BMC KCS Initialized at 0x%04x\n",kcsBaseAddress); + } + return STATUS_OK; +} + +int +SendTimedImbpRequest_ssif ( IMBPREQUESTDATA *requestData, + unsigned int timeout, UINT8 *resp_data, int *respDataLen, + unsigned char *compCode) +{ /* SendTimedImb for SMBus */ + unsigned char rq[IPMI_REQBUF_SIZE+35] = {0,}; /*SIZE + MAX_ISA_LENGTH=35*/ + unsigned char rp[IPMI_RSPBUF_SIZE+35] = {0,}; /*SIZE + MAX_ISA_LENGTH=35*/ + unsigned int i, rpl=0; + int status; + int respMax, rlen; + + i = _IOPL(3); + rq[0] = ((requestData->netFn << 2) | (requestData->rsLun & 0x03)); + rq[1] = requestData->cmdType; + if (sizeof(rq) < (requestData->dataLength + 2)) return(LAN_ERR_BADLENGTH); + for (i=0;i<=(unsigned int)requestData->dataLength;i++) + rq[i+2] = requestData->data[i]; + + respMax = *respDataLen; + if (respMax == 0) respMax = IPMI_RSPBUF_SIZE; + status = SendmBmcRequest(rq,requestData->dataLength+2,rp,&rpl,0); + if (status == IMB_SEND_ERROR) { + *respDataLen = 0; + return LAN_ERR_SEND_FAIL; + } + + if (rpl < 3) { + *respDataLen = 0; + *compCode = 0xCA; /*cannot return #bytes*/ + return LAN_ERR_TIMEOUT; + } else { + rlen = rpl-3; /* Chop off netfn/LUN , compcode, command */ + if (rlen > respMax) rlen = respMax; + *respDataLen = rlen; + *compCode = rp[2]; + for (i = 0; i < rlen; i++) resp_data[i] = rp[i+3]; + } + return ACCESS_OK; +} + +int SendmBmcRequest ( + unsigned char* request, + UINT32 reqLength, + unsigned char* response, + UINT32 * respLength, + UINT32 ipmiOldFlag + ) +{ + int retries = 5; + + /* Send Request - Retry 5 times at most */ + do{ + if (SendRequest(request, reqLength, &SMBChar) == 0) + break; + } while (0 < retries--); + + if (retries <= 0) { return IMB_SEND_ERROR; } + + /* Read Response - Retry 5 times at most */ + retries = 5; + do{ + if ((*respLength = ReadResponse(response, &SMBChar))!=(-1)) { + break; // Success + } + }while (0 < retries--); + + if (retries <= 0) { return IMB_SEND_ERROR; } + + return IMB_SUCCESS; +} + +static int +SendRequest(unsigned char* req, int length, + struct SMBCharStruct* smbChar) +{ + UINT8 data = 0; + // unsigned char* msgBuf = req; + int i; + UINT8 status = 0; + + // Delay of 50 ms before request is sent + usleep(100000); + + // Handle Intel & ServerWorks Chipsets + // Clear all status bits, Host Status Register + switch (smbChar->Controller) { + // Intel Status Register + case INTEL_SMBC: + status = ICH_HST_STA_ALL_ERRS|ICH_HST_STA_INTR|ICH_HST_STA_BYTE_DONE_STS; + break; + // ServerWorks Status Register + case SW_SMBC: + status = ICH_HST_STA_ALL_ERRS|ICH_HST_STA_INTR; + break; + }// End of Switch + + // Status Register + WritePortUchar( (((smbChar->baseAddr))+ICH_HST_STA), status); + + // Block protocol, Host Control Register + WritePortUchar(((smbChar->baseAddr)+ICH_HST_CNT), (UINT8)(ICH_HST_CNT_SMB_CMD_BLOCK)); + // IPMI command, Host Command Register + WritePortUchar(((smbChar->baseAddr)+ICH_HST_CMD), (UINT8)(0x02)); + // Slave address, Host Address Register + WritePortUchar(((smbChar->baseAddr)+ICH_XMIT_SLVA), (UINT8)(mBMCADDR)); + // Block length, Host DATA0 Register + WritePortUchar(((smbChar->baseAddr)+ICH_D0), (UINT8)(length)); + // Initialize timer + + switch (smbChar->Controller) { + // Handle Intel Chipset + case INTEL_SMBC: + // the first byte + WritePortUchar(((smbChar->baseAddr)+ICH_BLOCK_DB), *(UINT8 *)((UINT8 *)req )); + // Block protocol and Start, Host Control Register + WritePortUchar(((smbChar->baseAddr)+ICH_HST_CNT), ICH_HST_CNT_START | ICH_HST_CNT_SMB_CMD_BLOCK); + // Send byte by byte + for ( i = 1; i < length; i++ ) { + // OsSetTimer(&SendReqTimer); * Start timer * + do { + // Read host status register + ReadPortUchar((smbChar->baseAddr+ICH_HST_STA), &data); + // if (OsTimerTimedout(&SendReqTimer)) return -1; + // Any error or Interrupt bit or byte done bit set ? + } while ((data & status) == 0); + // OsCancelTimer(&SendReqTimer); * End Timer * + // Check for byte completion in block transfer + if ((data & ICH_HST_STA_BYTE_DONE_STS) != ICH_HST_STA_BYTE_DONE_STS) + break; + // Write next byte + WritePortUchar(((smbChar->baseAddr)+ICH_BLOCK_DB), *(UINT8 *)((UINT8 *)req + i)); + // Clear status bits, Host Status Register + WritePortUchar((smbChar->baseAddr), (UINT8)(data & status)); + } // End of for - Send byte by byte + break; // End of Intel Chipset Handling + // Handle ServerWorks Chipset + case SW_SMBC: + // Block length, Host DATA1 Register + WritePortUchar(((smbChar->baseAddr)+ICH_D1), (UINT8)(length)); + // Reset block inex + ReadPortUchar(((smbChar->baseAddr)+ICH_HST_CNT), &data); + // Put all bytes of the message to Block Data Register + for ( i = 0; i < length; i++ ) + WritePortUchar(((smbChar->baseAddr)+ICH_BLOCK_DB), *(UINT8 *)((UINT8 *)req + i)); + // Block protocol and Start, Host Control Register + WritePortUchar(((smbChar->baseAddr)+ICH_HST_CNT), ICH_HST_CNT_START | ICH_HST_CNT_SMB_CMD_BLOCK); + // OsSetTimer(&SendReqTimer); * Start timer * + do { + // Read host status register + ReadPortUchar((smbChar->baseAddr+ICH_HST_STA), &data); + // if (OsTimerTimedout(&SendReqTimer)) return -1; + } while ((data & status) == 0); // Any error or Interrupt bit set ? + // OsCancelTimer(&SendReqTimer); * End Timer * + break; // End of ServerWorks Chipset Handling + } // End of Switch Controller + + // Clear status bits, Host Status Register + WritePortUchar((smbChar->baseAddr), (UINT8)(data & status)); + if (data & ICH_HST_STA_ALL_ERRS) return -1; + + // Success + return 0; +} + +static int +ReadResponse( unsigned char* resp, + struct SMBCharStruct * smbChar) +{ + UINT8 data, dummy, length, status = 0; + UINT8 * msgBuf = (UINT8 *) resp; + int i; //, timeout =0; + // os_timer_t SendReqTimer; + + // Delay current thread - time to delay in uSecs + // 100 msec = 100000 microsec + usleep(100000); + + // Handle Intel & ServerWorks Chipsets + switch (smbChar->Controller) { + case INTEL_SMBC: + status = ICH_HST_STA_ALL_ERRS|ICH_HST_STA_INTR|ICH_HST_STA_BYTE_DONE_STS; + break; + case SW_SMBC: + status = ICH_HST_STA_ALL_ERRS|ICH_HST_STA_INTR; + break; + } // End of Switch + + WritePortUchar(((smbChar->baseAddr)+ICH_HST_STA), status); + // Block protocol, Host Control Register + WritePortUchar(((smbChar->baseAddr)+ICH_HST_CNT), ICH_HST_CNT_SMB_CMD_BLOCK); + // Slave address, Host Address Register + WritePortUchar(((smbChar->baseAddr)+ICH_XMIT_SLVA), (UINT8)(mBMCADDR |ICH_XMIT_SLVA_READ)); + // IPMI command, Host Command Register + WritePortUchar(((smbChar->baseAddr)+ICH_HST_CMD), 0x03); + // Reset block index + ReadPortUchar(((smbChar->baseAddr)+ICH_HST_CNT), &data); + // Block protocol and Start, Host Control Register + WritePortUchar(((smbChar->baseAddr)+ICH_HST_CNT), ICH_HST_CNT_START | ICH_HST_CNT_SMB_CMD_BLOCK); + + // Initialize timer + do { + // Read host status register + ReadPortUchar((smbChar->baseAddr+ICH_HST_STA), &data); + } while ((data & status) == 0); // Any error or Interrupt bit set ? + // End Timer + + // Clear status bits, Host Status Register + WritePortUchar((smbChar->baseAddr), (UINT8)(data & status)); + + if (data & ICH_HST_STA_ALL_ERRS) return -1; + + // Block length, Host DATA0 Register + ReadPortUchar(((smbChar->baseAddr)+ICH_D0), &length); + /* check recv length > MAX (35) */ + if (length > MAX_ISA_LENGTH) length = MAX_ISA_LENGTH; + switch (smbChar->Controller) { + case INTEL_SMBC: + // Read the first byte + ReadPortUchar((smbChar->baseAddr+ICH_BLOCK_DB), &(msgBuf[0])); + // Put all bytes of the message to Block Data Register + for ( i = 1; i < length; i++ ) { + if (i == (length-1)) { + // Set Last Byte bit + ReadPortUchar(((smbChar->baseAddr)+ICH_HST_CNT),&dummy); + WritePortUchar(((smbChar->baseAddr)+ICH_HST_CNT), (UINT8)(dummy | ICH_HST_CNT_LAST_BYTE)); + } + // Clear status bits, Host Status Register + WritePortUchar((smbChar->baseAddr), (UINT8)(data & status)); + + do { + // Read host status register + ReadPortUchar((smbChar->baseAddr+ICH_HST_STA), &data); + } while ((data & status) == 0); // Any error or Interrupt bit set ? + + if (data & ICH_HST_STA_ALL_ERRS) return -1; + ReadPortUchar((smbChar->baseAddr+ICH_BLOCK_DB), &(msgBuf[i])); + } + break; + case SW_SMBC: + // Put all bytes of the message to Block Data Register + for ( i = 0; i < length; i++ ) { + ReadPortUchar((smbChar->baseAddr+ICH_BLOCK_DB), &(msgBuf[i])); + } + break; + } /* end of switch */ + + if (data & ICH_HST_STA_ALL_ERRS) return -1; + + return length; +} +#endif + +/* end ipmidir.c */ |