Staging: add heci driver

The Intel Management Engine Interface (aka HECI: Host Embedded
Controller Interface ) enables communication between the host OS and
the Management Engine firmware. MEI is bi-directional, and either the
host or Intel AMT firmware can initiate transactions.

The core hardware architecture of Intel Active Management Technology
(Intel AMT) is resident in firmware. The micro-controller within the
chipset's graphics and memory controller (GMCH) hub houses the
Management Engine (ME) firmware, which implements various services
on behalf of management applications.

Some of the ME subsystems that can be access via MEI driver:

- Intel(R) Quiet System Technology (QST) is implemented as a firmware
subsystem  that  runs in the ME.  Programs that wish to expose the
health monitoring and fan speed control capabilities of Intel(R) QST
will need to use the MEI driver to communicate with the ME sub-system.
- ASF is the "Alert Standard Format" which is an DMTF manageability
standard. It is implemented in the PC's hardware and firmware, and is
managed from a remote console.

Most recent Intel desktop chipsets have one or more of the above ME
services. The MEI driver will make it possible to support the above
features on Linux and provides applications access to the ME and it's
features.

Signed-off-by: Anas Nashif <anas.nashif@intel.com>
Signed-off-by: Marcin Obara <marcin.obara@intel.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@suse.de>
This commit is contained in:
Marcin Obara
2009-02-25 12:29:24 -08:00
committed by Greg Kroah-Hartman
parent 2418a628ff
commit d52b3d9c72
14 changed files with 6627 additions and 0 deletions
+2
View File
@@ -107,5 +107,7 @@ source "drivers/staging/phison/Kconfig"
source "drivers/staging/p9auth/Kconfig"
source "drivers/staging/heci/Kconfig"
endif # !STAGING_EXCLUDE_BUILD
endif # STAGING
+1
View File
@@ -36,3 +36,4 @@ obj-$(CONFIG_USB_SERIAL_ATEN2011) += uc2322/
obj-$(CONFIG_B3DFG) += b3dfg/
obj-$(CONFIG_IDE_PHISON) += phison/
obj-$(CONFIG_PLAN9AUTH) += p9auth/
obj-$(CONFIG_HECI) += heci/
+6
View File
@@ -0,0 +1,6 @@
config HECI
tristate "Intel Management Engine Interface (MEI) Support"
---help---
The Intel Management Engine Interface (Intel MEI) driver allows
applications to access the Active Management Technology
firmware and other Management Engine sub-systems.
+9
View File
@@ -0,0 +1,9 @@
obj-$(CONFIG_HECI) += heci.o
heci-objs := \
heci_init.o \
interrupt.o \
heci_interface.o \
io_heci.o \
heci_main.o
+176
View File
@@ -0,0 +1,176 @@
/*
* Part of Intel(R) Manageability Engine Interface Linux driver
*
* Copyright (c) 2003 - 2008 Intel Corp.
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions, and the following disclaimer,
* without modification.
* 2. Redistributions in binary form must reproduce at minimum a disclaimer
* substantially similar to the "NO WARRANTY" disclaimer below
* ("Disclaimer") and any redistribution must be conditioned upon
* including a substantially similar Disclaimer requirement for further
* binary redistribution.
* 3. Neither the names of the above-listed copyright holders nor the names
* of any contributors may be used to endorse or promote products derived
* from this software without specific prior written permission.
*
* Alternatively, this software may be distributed under the terms of the
* GNU General Public License ("GPL") version 2 as published by the Free
* Software Foundation.
*
* NO WARRANTY
* 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 MERCHANTIBILITY AND FITNESS FOR
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
* HOLDERS OR CONTRIBUTORS BE LIABLE FOR 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 DAMAGES.
*
*/
#ifndef _HECI_H_
#define _HECI_H_
#include <linux/version.h>
#include <linux/spinlock.h>
#include <linux/list.h>
#include <linux/pci.h>
#include <linux/timer.h>
#include <linux/interrupt.h>
#include <linux/workqueue.h>
#include <linux/module.h>
#include <linux/aio.h>
#include <linux/types.h>
#include "heci_data_structures.h"
extern const struct guid heci_pthi_guid;
extern const struct guid heci_wd_guid;
extern const __u8 heci_start_wd_params[];
extern const __u8 heci_stop_wd_params[];
extern const __u8 heci_wd_state_independence_msg[3][4];
/*
* heci device ID
*/
#define HECI_DEV_ID_82946GZ 0x2974 /* 82946GZ/GL */
#define HECI_DEV_ID_82G35 0x2984 /* 82G35 Express */
#define HECI_DEV_ID_82Q965 0x2994 /* 82Q963/Q965 */
#define HECI_DEV_ID_82G965 0x29A4 /* 82P965/G965 */
#define HECI_DEV_ID_82GM965 0x2A04 /* Mobile PM965/GM965 */
#define HECI_DEV_ID_82GME965 0x2A14 /* Mobile GME965/GLE960 */
#define HECI_DEV_ID_ICH9_82Q35 0x29B4 /* 82Q35 Express */
#define HECI_DEV_ID_ICH9_82G33 0x29C4 /* 82G33/G31/P35/P31 Express */
#define HECI_DEV_ID_ICH9_82Q33 0x29D4 /* 82Q33 Express */
#define HECI_DEV_ID_ICH9_82X38 0x29E4 /* 82X38/X48 Express */
#define HECI_DEV_ID_ICH9_3200 0x29F4 /* 3200/3210 Server */
#define HECI_DEV_ID_ICH9_6 0x28B4 /* Bearlake */
#define HECI_DEV_ID_ICH9_7 0x28C4 /* Bearlake */
#define HECI_DEV_ID_ICH9_8 0x28D4 /* Bearlake */
#define HECI_DEV_ID_ICH9_9 0x28E4 /* Bearlake */
#define HECI_DEV_ID_ICH9_10 0x28F4 /* Bearlake */
#define HECI_DEV_ID_ICH9M_1 0x2A44 /* Cantiga */
#define HECI_DEV_ID_ICH9M_2 0x2A54 /* Cantiga */
#define HECI_DEV_ID_ICH9M_3 0x2A64 /* Cantiga */
#define HECI_DEV_ID_ICH9M_4 0x2A74 /* Cantiga */
#define HECI_DEV_ID_ICH10_1 0x2E04 /* Eaglelake */
#define HECI_DEV_ID_ICH10_2 0x2E14 /* Eaglelake */
#define HECI_DEV_ID_ICH10_3 0x2E24 /* Eaglelake */
#define HECI_DEV_ID_ICH10_4 0x2E34 /* Eaglelake */
/*
* heci init function prototypes
*/
struct iamt_heci_device *init_heci_device(struct pci_dev *pdev);
void heci_reset(struct iamt_heci_device *dev, int interrupts);
int heci_hw_init(struct iamt_heci_device *dev);
int heci_task_initialize_clients(void *data);
int heci_initialize_clients(struct iamt_heci_device *dev);
struct heci_file_private *heci_alloc_file_private(struct file *file);
int heci_disconnect_host_client(struct iamt_heci_device *dev,
struct heci_file_private *file_ext);
void heci_initialize_list(struct io_heci_list *list,
struct iamt_heci_device *dev);
void heci_flush_list(struct io_heci_list *list,
struct heci_file_private *file_ext);
void heci_flush_queues(struct iamt_heci_device *dev,
struct heci_file_private *file_ext);
void heci_remove_client_from_file_list(struct iamt_heci_device *dev,
__u8 host_client_id);
/*
* interrupt function prototype
*/
irqreturn_t heci_isr_interrupt(int irq, void *dev_id);
void heci_wd_timer(unsigned long data);
/*
* input output function prototype
*/
int heci_ioctl_get_version(struct iamt_heci_device *dev, int if_num,
struct heci_message_data *u_msg,
struct heci_message_data k_msg,
struct heci_file_private *file_ext);
int heci_ioctl_connect_client(struct iamt_heci_device *dev, int if_num,
struct heci_message_data *u_msg,
struct heci_message_data k_msg,
struct file *file);
int heci_ioctl_wd(struct iamt_heci_device *dev, int if_num,
struct heci_message_data k_msg,
struct heci_file_private *file_ext);
int heci_ioctl_bypass_wd(struct iamt_heci_device *dev, int if_num,
struct heci_message_data k_msg,
struct heci_file_private *file_ext);
int heci_start_read(struct iamt_heci_device *dev, int if_num,
struct heci_file_private *file_ext);
int pthi_write(struct iamt_heci_device *dev,
struct heci_cb_private *priv_cb);
int pthi_read(struct iamt_heci_device *dev, int if_num, struct file *file,
char *ubuf, size_t length, loff_t *offset);
struct heci_cb_private *find_pthi_read_list_entry(
struct iamt_heci_device *dev,
struct file *file);
void run_next_iamthif_cmd(struct iamt_heci_device *dev);
void heci_free_cb_private(struct heci_cb_private *priv_cb);
/**
* heci_fe_same_id - tell if file private data have same id
*
* @fe1: private data of 1. file object
* @fe2: private data of 2. file object
*
* returns !=0 - if ids are the same, 0 - if differ.
*/
static inline int heci_fe_same_id(const struct heci_file_private *fe1,
const struct heci_file_private *fe2)
{
return ((fe1->host_client_id == fe2->host_client_id)
&& (fe1->me_client_id == fe2->me_client_id));
}
#endif /* _HECI_H_ */
File diff suppressed because it is too large Load Diff
File diff suppressed because it is too large Load Diff
+485
View File
@@ -0,0 +1,485 @@
/*
* Part of Intel(R) Manageability Engine Interface Linux driver
*
* Copyright (c) 2003 - 2008 Intel Corp.
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions, and the following disclaimer,
* without modification.
* 2. Redistributions in binary form must reproduce at minimum a disclaimer
* substantially similar to the "NO WARRANTY" disclaimer below
* ("Disclaimer") and any redistribution must be conditioned upon
* including a substantially similar Disclaimer requirement for further
* binary redistribution.
* 3. Neither the names of the above-listed copyright holders nor the names
* of any contributors may be used to endorse or promote products derived
* from this software without specific prior written permission.
*
* Alternatively, this software may be distributed under the terms of the
* GNU General Public License ("GPL") version 2 as published by the Free
* Software Foundation.
*
* NO WARRANTY
* 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 MERCHANTIBILITY AND FITNESS FOR
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
* HOLDERS OR CONTRIBUTORS BE LIABLE FOR 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 DAMAGES.
*
*/
#include "heci.h"
#include "heci_interface.h"
/**
* heci_set_csr_register - write H_CSR register to the heci device
*
* @dev: device object for our driver
*/
void heci_set_csr_register(struct iamt_heci_device *dev)
{
write_heci_register(dev, H_CSR, dev->host_hw_state);
dev->host_hw_state = read_heci_register(dev, H_CSR);
}
/**
* heci_csr_enable_interrupts - enable heci device interrupts
*
* @dev: device object for our driver
*/
void heci_csr_enable_interrupts(struct iamt_heci_device *dev)
{
dev->host_hw_state |= H_IE;
heci_set_csr_register(dev);
}
/**
* heci_csr_disable_interrupts - disable heci device interrupts
*
* @dev: device object for our driver
*/
void heci_csr_disable_interrupts(struct iamt_heci_device *dev)
{
dev->host_hw_state &= ~H_IE;
heci_set_csr_register(dev);
}
/**
* _host_get_filled_slots - get number of device filled buffer slots
*
* @device: the device structure
*
* returns numer of filled slots
*/
static unsigned char _host_get_filled_slots(const struct iamt_heci_device *dev)
{
char read_ptr, write_ptr;
read_ptr = (char) ((dev->host_hw_state & H_CBRP) >> 8);
write_ptr = (char) ((dev->host_hw_state & H_CBWP) >> 16);
return (unsigned char) (write_ptr - read_ptr);
}
/**
* host_buffer_is_empty - check if host buffer is empty.
*
* @dev: device object for our driver
*
* returns 1 if empty, 0 - otherwise.
*/
int host_buffer_is_empty(struct iamt_heci_device *dev)
{
unsigned char filled_slots;
dev->host_hw_state = read_heci_register(dev, H_CSR);
filled_slots = _host_get_filled_slots(dev);
if (filled_slots > 0)
return 0;
return 1;
}
/**
* count_empty_write_slots - count write empty slots.
*
* @dev: device object for our driver
*
* returns -1(ESLOTS_OVERFLOW) if overflow, otherwise empty slots count
*/
__s32 count_empty_write_slots(const struct iamt_heci_device *dev)
{
unsigned char buffer_depth, filled_slots, empty_slots;
buffer_depth = (unsigned char) ((dev->host_hw_state & H_CBD) >> 24);
filled_slots = _host_get_filled_slots(dev);
empty_slots = buffer_depth - filled_slots;
if (filled_slots > buffer_depth) {
/* overflow */
return -ESLOTS_OVERFLOW;
}
return (__s32) empty_slots;
}
/**
* heci_write_message - write a message to heci device.
*
* @dev: device object for our driver
* @heci_hdr: header of message
* @write_buffer: message buffer will be write
* @write_length: message size will be write
*
* returns 1 if success, 0 - otherwise.
*/
int heci_write_message(struct iamt_heci_device *dev,
struct heci_msg_hdr *header,
unsigned char *write_buffer,
unsigned long write_length)
{
__u32 temp_msg = 0;
unsigned long bytes_written = 0;
unsigned char buffer_depth, filled_slots, empty_slots;
unsigned long dw_to_write;
dev->host_hw_state = read_heci_register(dev, H_CSR);
DBG("host_hw_state = 0x%08x.\n", dev->host_hw_state);
DBG("heci_write_message header=%08x.\n", *((__u32 *) header));
buffer_depth = (unsigned char) ((dev->host_hw_state & H_CBD) >> 24);
filled_slots = _host_get_filled_slots(dev);
empty_slots = buffer_depth - filled_slots;
DBG("filled = %hu, empty = %hu.\n", filled_slots, empty_slots);
dw_to_write = ((write_length + 3) / 4);
if (dw_to_write > empty_slots)
return 0;
write_heci_register(dev, H_CB_WW, *((__u32 *) header));
while (write_length >= 4) {
write_heci_register(dev, H_CB_WW,
*(__u32 *) (write_buffer + bytes_written));
bytes_written += 4;
write_length -= 4;
}
if (write_length > 0) {
memcpy(&temp_msg, &write_buffer[bytes_written], write_length);
write_heci_register(dev, H_CB_WW, temp_msg);
}
dev->host_hw_state |= H_IG;
write_heci_register(dev, H_CSR, dev->host_hw_state);
dev->me_hw_state = read_heci_register(dev, ME_CSR_HA);
if ((dev->me_hw_state & ME_RDY_HRA) != ME_RDY_HRA)
return 0;
dev->write_hang = 0;
return 1;
}
/**
* count_full_read_slots - count read full slots.
*
* @dev: device object for our driver
*
* returns -1(ESLOTS_OVERFLOW) if overflow, otherwise filled slots count
*/
__s32 count_full_read_slots(struct iamt_heci_device *dev)
{
char read_ptr, write_ptr;
unsigned char buffer_depth, filled_slots;
dev->me_hw_state = read_heci_register(dev, ME_CSR_HA);
buffer_depth = (unsigned char)((dev->me_hw_state & ME_CBD_HRA) >> 24);
read_ptr = (char) ((dev->me_hw_state & ME_CBRP_HRA) >> 8);
write_ptr = (char) ((dev->me_hw_state & ME_CBWP_HRA) >> 16);
filled_slots = (unsigned char) (write_ptr - read_ptr);
if (filled_slots > buffer_depth) {
/* overflow */
return -ESLOTS_OVERFLOW;
}
DBG("filled_slots =%08x \n", filled_slots);
return (__s32) filled_slots;
}
/**
* heci_read_slots - read a message from heci device.
*
* @dev: device object for our driver
* @buffer: message buffer will be write
* @buffer_length: message size will be read
*/
void heci_read_slots(struct iamt_heci_device *dev,
unsigned char *buffer, unsigned long buffer_length)
{
__u32 i = 0;
unsigned char temp_buf[sizeof(__u32)];
while (buffer_length >= sizeof(__u32)) {
((__u32 *) buffer)[i] = read_heci_register(dev, ME_CB_RW);
DBG("buffer[%d]= %d\n", i, ((__u32 *) buffer)[i]);
i++;
buffer_length -= sizeof(__u32);
}
if (buffer_length > 0) {
*((__u32 *) &temp_buf) = read_heci_register(dev, ME_CB_RW);
memcpy(&buffer[i * 4], temp_buf, buffer_length);
}
dev->host_hw_state |= H_IG;
heci_set_csr_register(dev);
}
/**
* flow_ctrl_creds - check flow_control credentials.
*
* @dev: device object for our driver
* @file_ext: private data of the file object
*
* returns 1 if flow_ctrl_creds >0, 0 - otherwise.
*/
int flow_ctrl_creds(struct iamt_heci_device *dev,
struct heci_file_private *file_ext)
{
__u8 i;
if (!dev->num_heci_me_clients)
return 0;
if (file_ext == NULL)
return 0;
if (file_ext->flow_ctrl_creds > 0)
return 1;
for (i = 0; i < dev->num_heci_me_clients; i++) {
if (dev->me_clients[i].client_id == file_ext->me_client_id) {
if (dev->me_clients[i].flow_ctrl_creds > 0) {
BUG_ON(dev->me_clients[i].props.single_recv_buf
== 0);
return 1;
}
return 0;
}
}
BUG();
return 0;
}
/**
* flow_ctrl_reduce - reduce flow_control.
*
* @dev: device object for our driver
* @file_ext: private data of the file object
*/
void flow_ctrl_reduce(struct iamt_heci_device *dev,
struct heci_file_private *file_ext)
{
__u8 i;
if (!dev->num_heci_me_clients)
return;
for (i = 0; i < dev->num_heci_me_clients; i++) {
if (dev->me_clients[i].client_id == file_ext->me_client_id) {
if (dev->me_clients[i].props.single_recv_buf != 0) {
BUG_ON(dev->me_clients[i].flow_ctrl_creds <= 0);
dev->me_clients[i].flow_ctrl_creds--;
} else {
BUG_ON(file_ext->flow_ctrl_creds <= 0);
file_ext->flow_ctrl_creds--;
}
return;
}
}
BUG();
}
/**
* heci_send_flow_control - send flow control to fw.
*
* @dev: device object for our driver
* @file_ext: private data of the file object
*
* returns 1 if success, 0 - otherwise.
*/
int heci_send_flow_control(struct iamt_heci_device *dev,
struct heci_file_private *file_ext)
{
struct heci_msg_hdr *heci_hdr;
struct hbm_flow_control *heci_flow_control;
heci_hdr = (struct heci_msg_hdr *) &dev->wr_msg_buf[0];
heci_hdr->host_addr = 0;
heci_hdr->me_addr = 0;
heci_hdr->length = sizeof(struct hbm_flow_control);
heci_hdr->msg_complete = 1;
heci_hdr->reserved = 0;
heci_flow_control = (struct hbm_flow_control *) &dev->wr_msg_buf[1];
memset(heci_flow_control, 0, sizeof(heci_flow_control));
heci_flow_control->host_addr = file_ext->host_client_id;
heci_flow_control->me_addr = file_ext->me_client_id;
heci_flow_control->cmd.cmd = HECI_FLOW_CONTROL_CMD;
memset(heci_flow_control->reserved, 0,
sizeof(heci_flow_control->reserved));
DBG("sending flow control host client = %d, me client = %d\n",
file_ext->host_client_id, file_ext->me_client_id);
if (!heci_write_message(dev, heci_hdr,
(unsigned char *) heci_flow_control,
sizeof(struct hbm_flow_control)))
return 0;
return 1;
}
/**
* other_client_is_connecting - check if other
* client with the same client id is connected.
*
* @dev: device object for our driver
* @file_ext: private data of the file object
*
* returns 1 if other client is connected, 0 - otherwise.
*/
int other_client_is_connecting(struct iamt_heci_device *dev,
struct heci_file_private *file_ext)
{
struct heci_file_private *file_pos = NULL;
struct heci_file_private *file_next = NULL;
list_for_each_entry_safe(file_pos, file_next, &dev->file_list, link) {
if ((file_pos->state == HECI_FILE_CONNECTING)
&& (file_pos != file_ext)
&& file_ext->me_client_id == file_pos->me_client_id)
return 1;
}
return 0;
}
/**
* heci_send_wd - send watch dog message to fw.
*
* @dev: device object for our driver
*
* returns 1 if success, 0 - otherwise.
*/
int heci_send_wd(struct iamt_heci_device *dev)
{
struct heci_msg_hdr *heci_hdr;
heci_hdr = (struct heci_msg_hdr *) &dev->wr_msg_buf[0];
heci_hdr->host_addr = dev->wd_file_ext.host_client_id;
heci_hdr->me_addr = dev->wd_file_ext.me_client_id;
heci_hdr->msg_complete = 1;
heci_hdr->reserved = 0;
if (!memcmp(dev->wd_data, heci_start_wd_params,
HECI_WD_PARAMS_SIZE)) {
heci_hdr->length = HECI_START_WD_DATA_SIZE;
} else {
BUG_ON(memcmp(dev->wd_data, heci_stop_wd_params,
HECI_WD_PARAMS_SIZE));
heci_hdr->length = HECI_WD_PARAMS_SIZE;
}
if (!heci_write_message(dev, heci_hdr, dev->wd_data, heci_hdr->length))
return 0;
return 1;
}
/**
* heci_disconnect - send disconnect message to fw.
*
* @dev: device object for our driver
* @file_ext: private data of the file object
*
* returns 1 if success, 0 - otherwise.
*/
int heci_disconnect(struct iamt_heci_device *dev,
struct heci_file_private *file_ext)
{
struct heci_msg_hdr *heci_hdr;
struct hbm_client_disconnect_request *heci_cli_disconnect;
heci_hdr = (struct heci_msg_hdr *) &dev->wr_msg_buf[0];
heci_hdr->host_addr = 0;
heci_hdr->me_addr = 0;
heci_hdr->length = sizeof(struct hbm_client_disconnect_request);
heci_hdr->msg_complete = 1;
heci_hdr->reserved = 0;
heci_cli_disconnect =
(struct hbm_client_disconnect_request *) &dev->wr_msg_buf[1];
memset(heci_cli_disconnect, 0, sizeof(heci_cli_disconnect));
heci_cli_disconnect->host_addr = file_ext->host_client_id;
heci_cli_disconnect->me_addr = file_ext->me_client_id;
heci_cli_disconnect->cmd.cmd = CLIENT_DISCONNECT_REQ_CMD;
heci_cli_disconnect->reserved[0] = 0;
if (!heci_write_message(dev, heci_hdr,
(unsigned char *) heci_cli_disconnect,
sizeof(struct hbm_client_disconnect_request)))
return 0;
return 1;
}
/**
* heci_connect - send connect message to fw.
*
* @dev: device object for our driver
* @file_ext: private data of the file object
*
* returns 1 if success, 0 - otherwise.
*/
int heci_connect(struct iamt_heci_device *dev,
struct heci_file_private *file_ext)
{
struct heci_msg_hdr *heci_hdr;
struct hbm_client_connect_request *heci_cli_connect;
heci_hdr = (struct heci_msg_hdr *) &dev->wr_msg_buf[0];
heci_hdr->host_addr = 0;
heci_hdr->me_addr = 0;
heci_hdr->length = sizeof(struct hbm_client_connect_request);
heci_hdr->msg_complete = 1;
heci_hdr->reserved = 0;
heci_cli_connect =
(struct hbm_client_connect_request *) &dev->wr_msg_buf[1];
heci_cli_connect->host_addr = file_ext->host_client_id;
heci_cli_connect->me_addr = file_ext->me_client_id;
heci_cli_connect->cmd.cmd = CLIENT_CONNECT_REQ_CMD;
heci_cli_connect->reserved = 0;
if (!heci_write_message(dev, heci_hdr,
(unsigned char *) heci_cli_connect,
sizeof(struct hbm_client_connect_request)))
return 0;
return 1;
}
+170
View File
@@ -0,0 +1,170 @@
/*
* Part of Intel(R) Manageability Engine Interface Linux driver
*
* Copyright (c) 2003 - 2008 Intel Corp.
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions, and the following disclaimer,
* without modification.
* 2. Redistributions in binary form must reproduce at minimum a disclaimer
* substantially similar to the "NO WARRANTY" disclaimer below
* ("Disclaimer") and any redistribution must be conditioned upon
* including a substantially similar Disclaimer requirement for further
* binary redistribution.
* 3. Neither the names of the above-listed copyright holders nor the names
* of any contributors may be used to endorse or promote products derived
* from this software without specific prior written permission.
*
* Alternatively, this software may be distributed under the terms of the
* GNU General Public License ("GPL") version 2 as published by the Free
* Software Foundation.
*
* NO WARRANTY
* 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 MERCHANTIBILITY AND FITNESS FOR
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
* HOLDERS OR CONTRIBUTORS BE LIABLE FOR 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 DAMAGES.
*
*/
#ifndef _HECI_INTERFACE_H_
#define _HECI_INTERFACE_H_
#include <linux/version.h>
#include <linux/spinlock.h>
#include <linux/list.h>
#include <linux/pci.h>
#include <linux/timer.h>
#include <linux/interrupt.h>
#include <linux/workqueue.h>
#include <linux/module.h>
#include <linux/aio.h>
#include <linux/types.h>
#include "heci_data_structures.h"
#define HBM_MINOR_VERSION 0
#define HBM_MAJOR_VERSION 1
#define HBM_TIMEOUT 1 /* 1 second */
#define HOST_START_REQ_CMD 0x01
#define HOST_START_RES_CMD 0x81
#define HOST_STOP_REQ_CMD 0x02
#define HOST_STOP_RES_CMD 0x82
#define ME_STOP_REQ_CMD 0x03
#define HOST_ENUM_REQ_CMD 0x04
#define HOST_ENUM_RES_CMD 0x84
#define HOST_CLIENT_PROPERTEIS_REQ_CMD 0x05
#define HOST_CLIENT_PROPERTEIS_RES_CMD 0x85
#define CLIENT_CONNECT_REQ_CMD 0x06
#define CLIENT_CONNECT_RES_CMD 0x86
#define CLIENT_DISCONNECT_REQ_CMD 0x07
#define CLIENT_DISCONNECT_RES_CMD 0x87
#define HECI_FLOW_CONTROL_CMD 0x08
#define AMT_WD_VALUE 120 /* seconds */
#define HECI_WATCHDOG_DATA_SIZE 16
#define HECI_START_WD_DATA_SIZE 20
#define HECI_WD_PARAMS_SIZE 4
/* IOCTL commands */
#define IOCTL_HECI_GET_VERSION \
_IOWR('H' , 0x0, struct heci_message_data)
#define IOCTL_HECI_CONNECT_CLIENT \
_IOWR('H' , 0x01, struct heci_message_data)
#define IOCTL_HECI_WD \
_IOWR('H' , 0x02, struct heci_message_data)
#define IOCTL_HECI_BYPASS_WD \
_IOWR('H' , 0x10, struct heci_message_data)
enum heci_stop_reason_types{
DRIVER_STOP_REQUEST = 0x00,
DEVICE_D1_ENTRY = 0x01,
DEVICE_D2_ENTRY = 0x02,
DEVICE_D3_ENTRY = 0x03,
SYSTEM_S1_ENTRY = 0x04,
SYSTEM_S2_ENTRY = 0x05,
SYSTEM_S3_ENTRY = 0x06,
SYSTEM_S4_ENTRY = 0x07,
SYSTEM_S5_ENTRY = 0x08
};
enum me_stop_reason_types{
FW_UPDATE = 0x00
};
enum client_connect_status_types{
CCS_SUCCESS = 0x00,
CCS_NOT_FOUND = 0x01,
CCS_ALREADY_STARTED = 0x02,
CCS_OUT_OF_RESOURCES = 0x03,
CCS_MESSAGE_SMALL = 0x04
};
enum client_disconnect_status_types{
CDS_SUCCESS = 0x00
};
/*
* heci interface function prototypes
*/
void heci_set_csr_register(struct iamt_heci_device *dev);
void heci_csr_enable_interrupts(struct iamt_heci_device *dev);
void heci_csr_disable_interrupts(struct iamt_heci_device *dev);
void heci_read_slots(struct iamt_heci_device *dev,
unsigned char *buffer, unsigned long buffer_length);
int heci_write_message(struct iamt_heci_device *dev,
struct heci_msg_hdr *header,
unsigned char *write_buffer,
unsigned long write_length);
int host_buffer_is_empty(struct iamt_heci_device *dev);
__s32 count_full_read_slots(struct iamt_heci_device *dev);
__s32 count_empty_write_slots(const struct iamt_heci_device *dev);
int flow_ctrl_creds(struct iamt_heci_device *dev,
struct heci_file_private *file_ext);
int heci_send_wd(struct iamt_heci_device *dev);
void flow_ctrl_reduce(struct iamt_heci_device *dev,
struct heci_file_private *file_ext);
int heci_send_flow_control(struct iamt_heci_device *dev,
struct heci_file_private *file_ext);
int heci_disconnect(struct iamt_heci_device *dev,
struct heci_file_private *file_ext);
int other_client_is_connecting(struct iamt_heci_device *dev,
struct heci_file_private *file_ext);
int heci_connect(struct iamt_heci_device *dev,
struct heci_file_private *file_ext);
#endif /* _HECI_INTERFACE_H_ */
File diff suppressed because it is too large Load Diff
+54
View File
@@ -0,0 +1,54 @@
/*
* Part of Intel(R) Manageability Engine Interface Linux driver
*
* Copyright (c) 2003 - 2008 Intel Corp.
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions, and the following disclaimer,
* without modification.
* 2. Redistributions in binary form must reproduce at minimum a disclaimer
* substantially similar to the "NO WARRANTY" disclaimer below
* ("Disclaimer") and any redistribution must be conditioned upon
* including a substantially similar Disclaimer requirement for further
* binary redistribution.
* 3. Neither the names of the above-listed copyright holders nor the names
* of any contributors may be used to endorse or promote products derived
* from this software without specific prior written permission.
*
* Alternatively, this software may be distributed under the terms of the
* GNU General Public License ("GPL") version 2 as published by the Free
* Software Foundation.
*
* NO WARRANTY
* 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 MERCHANTIBILITY AND FITNESS FOR
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
* HOLDERS OR CONTRIBUTORS BE LIABLE FOR 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 DAMAGES.
*
*/
#ifndef HECI_VERSION_H
#define HECI_VERSION_H
#define MAJOR_VERSION 5
#define MINOR_VERSION 0
#define QUICK_FIX_NUMBER 0
#define VER_BUILD 31
#define HECI_DRV_VER1 __stringify(MAJOR_VERSION) "." __stringify(MINOR_VERSION)
#define HECI_DRV_VER2 __stringify(QUICK_FIX_NUMBER) "." __stringify(VER_BUILD)
#define HECI_DRIVER_VERSION HECI_DRV_VER1 "." HECI_DRV_VER2
#endif
File diff suppressed because it is too large Load Diff
File diff suppressed because it is too large Load Diff
+150
View File
@@ -0,0 +1,150 @@
/*
* Part of Intel(R) Manageability Engine Interface Linux driver
*
* Copyright (c) 2003 - 2008 Intel Corp.
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions, and the following disclaimer,
* without modification.
* 2. Redistributions in binary form must reproduce at minimum a disclaimer
* substantially similar to the "NO WARRANTY" disclaimer below
* ("Disclaimer") and any redistribution must be conditioned upon
* including a substantially similar Disclaimer requirement for further
* binary redistribution.
* 3. Neither the names of the above-listed copyright holders nor the names
* of any contributors may be used to endorse or promote products derived
* from this software without specific prior written permission.
*
* Alternatively, this software may be distributed under the terms of the
* GNU General Public License ("GPL") version 2 as published by the Free
* Software Foundation.
*
* NO WARRANTY
* 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 MERCHANTIBILITY AND FITNESS FOR
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
* HOLDERS OR CONTRIBUTORS BE LIABLE FOR 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 DAMAGES.
*
*/
#ifndef _KCOMPAT_H_
#define _KCOMPAT_H_
#include <linux/version.h>
#include <linux/init.h>
#include <linux/types.h>
#include <linux/errno.h>
#include <linux/module.h>
#include <linux/interrupt.h>
#include <asm/io.h>
#if (LINUX_VERSION_CODE < KERNEL_VERSION(2, 6, 27))
#if (LINUX_VERSION_CODE < KERNEL_VERSION(2, 6, 18))
#define device_create(cls, parent, devt, devpar, fmt...) class_device_create(cls, parent, devt, NULL, ## fmt)
#define HECI_DEVICE_CREATE(cls, parent, devt, devpar, fmt...) class_device_create(cls, parent, devt, NULL, ## fmt)
#define device_destroy class_device_destroy
#else
#define HECI_DEVICE_CREATE(cls, parent, devt, devpar, fmt...) device_create(cls, parent, devt, ## fmt)
#endif
#endif
#if (LINUX_VERSION_CODE < KERNEL_VERSION(2, 6, 23))
#define HECI_TASK_NONFREEZABLE current->flags |= PF_NOFREEZE;
#endif
#if (LINUX_VERSION_CODE < KERNEL_VERSION(2, 6, 17))
#include <asm/uaccess.h>
#else
#include <linux/uaccess.h>
#endif
/*****************************************************************************/
#if (LINUX_VERSION_CODE < KERNEL_VERSION(2, 6, 20))
#if (LINUX_VERSION_CODE >= KERNEL_VERSION(2, 5, 28))
#undef INIT_WORK
#define INIT_WORK(_work, _func) \
do { \
INIT_LIST_HEAD(&(_work)->entry); \
(_work)->pending = 0; \
(_work)->func = (void (*)(void *))_func; \
(_work)->data = _work; \
init_timer(&(_work)->timer); \
} while (0)
#undef PREPARE_WORK
#define PREPARE_WORK(_work, _func) \
do { \
(_work)->func = (void (*)(void *))_func; \
(_work)->data = _work; \
} while (0)
#endif
#ifndef round_jiffies
#define round_jiffies(x) x
#endif
#endif /* < 2.6.20 */
/*****************************************************************************/
#if (LINUX_VERSION_CODE < KERNEL_VERSION(2, 6, 18))
#ifndef IRQF_PROBE_SHARED
#ifdef SA_PROBEIRQ
#define IRQF_PROBE_SHARED SA_PROBEIRQ
#else
#define IRQF_PROBE_SHARED 0
#endif
#endif
#ifndef IRQF_SHARED
#define IRQF_SHARED SA_SHIRQ
#endif
#ifndef ARRAY_SIZE
#define ARRAY_SIZE(x) (sizeof(x) / sizeof((x)[0]))
#endif
#endif /* < 2.6.18 */
/*****************************************************************************/
#if (LINUX_VERSION_CODE < KERNEL_VERSION(2, 6, 19))
#if (!(defined(RHEL_MAJOR) && (RHEL_MAJOR == 5)))
#if (!(defined(RHEL_VERSION) && (RHEL_VERSION == 4) && (RHEL_UPDATE >= 5)))
typedef irqreturn_t (*irq_handler_t)(int, void*, struct pt_regs *);
#endif
#endif
typedef irqreturn_t (*new_handler_t)(int, void*);
static inline irqreturn_t _kc_request_irq(unsigned int irq,
new_handler_t handler,
unsigned long flags,
const char *devname,
void *dev_id)
{
irq_handler_t new_handler = (irq_handler_t) handler;
return request_irq(irq, new_handler, flags, devname, dev_id);
}
#undef request_irq
#define request_irq(irq, handler, flags, devname, dev_id) _kc_request_irq((irq), (handler), (flags), (devname), (dev_id))
#endif
#endif