Commit b84382f5 authored by Linus Torvalds's avatar Linus Torvalds

Merge tag 'char-misc-3.6-rc1' of git://git.kernel.org/pub/scm/linux/kernel/git/gregkh/char-misc

Pull char/misc patches from Greg Kroah-Hartman:
 "Here's the "big" pull request for 3.6-rc1 for the char/misc drivers.

  It's really just a few updates to the mei driver, plus 4 other tiny
  patches, nothing big at all.

  Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>"

* tag 'char-misc-3.6-rc1' of git://git.kernel.org/pub/scm/linux/kernel/git/gregkh/char-misc:
  mei: use module_pci_driver
  powerpc/BSR: cleanup the error path of bsr_init
  mei: mei_irq_thread_write_handler - line break fix
  mei: streamline the _mei_irq_thread_close/ioctol functions
  mei: introduce mei_data2slots wrapper
  mei: mei_wd_host_init: update the comment
  mei: remove write only wariable wd_due_counter
  mei: mei_device can be const for mei register access functions
  mei: revamp host buffer interface function
  mei: don't query HCSR for host buffer depth
  mei: group wd_interface_reg with watchdog variables within struct mei_device
  mei: mei_irq_thread_write_handler check for overflow
  mei: make mei_write_message more readable
  mei: check for error codes that mei_flow_ctrl_creds retuns
  misc: at25: Parse dt settings
  misc: hpilo: increase number of max supported channels
  mei: mei.txt: minor grammar fixes
parents fa93669a 6078188e
Atmel AT25 eeprom
Required properties:
- compatible : "atmel,at25".
- reg : chip select number
- spi-max-frequency : max spi frequency to use
- at25,byte-len : total eeprom size in bytes
- at25,addr-mode : addr-mode flags, as defined in include/linux/spi/eeprom.h
- at25,page-size : size of the eeprom page
Examples:
at25@0 {
compatible = "atmel,at25";
reg = <0>
spi-max-frequency = <5000000>;
at25,byte-len = <0x8000>;
at25,addr-mode = <2>;
at25,page-size = <64>;
};
...@@ -50,25 +50,25 @@ Intel MEI Driver ...@@ -50,25 +50,25 @@ Intel MEI Driver
The driver exposes a misc device called /dev/mei. The driver exposes a misc device called /dev/mei.
An application maintains communication with an Intel ME feature while An application maintains communication with an Intel ME feature while
/dev/mei is open. The binding to a specific features is performed by calling /dev/mei is open. The binding to a specific feature is performed by calling
MEI_CONNECT_CLIENT_IOCTL, which passes the desired UUID. MEI_CONNECT_CLIENT_IOCTL, which passes the desired UUID.
The number of instances of an Intel ME feature that can be opened The number of instances of an Intel ME feature that can be opened
at the same time depends on the Intel ME feature, but most of the at the same time depends on the Intel ME feature, but most of the
features allow only a single instance. features allow only a single instance.
The Intel AMT Host Interface (Intel AMTHI) feature supports multiple The Intel AMT Host Interface (Intel AMTHI) feature supports multiple
simultaneous user applications. Therefore, the Intel MEI driver handles simultaneous user connected applications. The Intel MEI driver
this internally by maintaining request queues for the applications. handles this internally by maintaining request queues for the applications.
The driver is oblivious to data that is passed between firmware feature The driver is transparent to data that are passed between firmware feature
and host application. and host application.
Because some of the Intel ME features can change the system Because some of the Intel ME features can change the system
configuration, the driver by default allows only a privileged configuration, the driver by default allows only a privileged
user to access it. user to access it.
A code snippet for an application communicating with A code snippet for an application communicating with Intel AMTHI client:
Intel AMTHI client:
struct mei_connect_client_data data; struct mei_connect_client_data data;
fd = open(MEI_DEVICE); fd = open(MEI_DEVICE);
...@@ -185,7 +185,7 @@ The Intel AMT Watchdog is composed of two parts: ...@@ -185,7 +185,7 @@ The Intel AMT Watchdog is composed of two parts:
2) Intel MEI driver - connects to the watchdog feature, configures the 2) Intel MEI driver - connects to the watchdog feature, configures the
watchdog and sends the heartbeats. watchdog and sends the heartbeats.
The Intel MEI driver uses the kernel watchdog to configure the Intel AMT The Intel MEI driver uses the kernel watchdog API to configure the Intel AMT
Watchdog and to send heartbeats to it. The default timeout of the Watchdog and to send heartbeats to it. The default timeout of the
watchdog is 120 seconds. watchdog is 120 seconds.
......
...@@ -297,7 +297,6 @@ static int __init bsr_init(void) ...@@ -297,7 +297,6 @@ static int __init bsr_init(void)
struct device_node *np; struct device_node *np;
dev_t bsr_dev; dev_t bsr_dev;
int ret = -ENODEV; int ret = -ENODEV;
int result;
np = of_find_compatible_node(NULL, NULL, "ibm,bsr"); np = of_find_compatible_node(NULL, NULL, "ibm,bsr");
if (!np) if (!np)
...@@ -306,13 +305,14 @@ static int __init bsr_init(void) ...@@ -306,13 +305,14 @@ static int __init bsr_init(void)
bsr_class = class_create(THIS_MODULE, "bsr"); bsr_class = class_create(THIS_MODULE, "bsr");
if (IS_ERR(bsr_class)) { if (IS_ERR(bsr_class)) {
printk(KERN_ERR "class_create() failed for bsr_class\n"); printk(KERN_ERR "class_create() failed for bsr_class\n");
ret = PTR_ERR(bsr_class);
goto out_err_1; goto out_err_1;
} }
bsr_class->dev_attrs = bsr_dev_attrs; bsr_class->dev_attrs = bsr_dev_attrs;
result = alloc_chrdev_region(&bsr_dev, 0, BSR_MAX_DEVS, "bsr"); ret = alloc_chrdev_region(&bsr_dev, 0, BSR_MAX_DEVS, "bsr");
bsr_major = MAJOR(bsr_dev); bsr_major = MAJOR(bsr_dev);
if (result < 0) { if (ret < 0) {
printk(KERN_ERR "alloc_chrdev_region() failed for bsr\n"); printk(KERN_ERR "alloc_chrdev_region() failed for bsr\n");
goto out_err_2; goto out_err_2;
} }
......
...@@ -19,7 +19,7 @@ ...@@ -19,7 +19,7 @@
#include <linux/spi/spi.h> #include <linux/spi/spi.h>
#include <linux/spi/eeprom.h> #include <linux/spi/eeprom.h>
#include <linux/of.h>
/* /*
* NOTE: this is an *EEPROM* driver. The vagaries of product naming * NOTE: this is an *EEPROM* driver. The vagaries of product naming
...@@ -305,25 +305,54 @@ static ssize_t at25_mem_write(struct memory_accessor *mem, const char *buf, ...@@ -305,25 +305,54 @@ static ssize_t at25_mem_write(struct memory_accessor *mem, const char *buf,
static int at25_probe(struct spi_device *spi) static int at25_probe(struct spi_device *spi)
{ {
struct at25_data *at25 = NULL; struct at25_data *at25 = NULL;
const struct spi_eeprom *chip; struct spi_eeprom chip;
struct device_node *np = spi->dev.of_node;
int err; int err;
int sr; int sr;
int addrlen; int addrlen;
/* Chip description */ /* Chip description */
chip = spi->dev.platform_data; if (!spi->dev.platform_data) {
if (!chip) { if (np) {
dev_dbg(&spi->dev, "no chip description\n"); u32 val;
err = -ENODEV;
goto fail; memset(&chip, 0, sizeof(chip));
} strncpy(chip.name, np->name, 10);
err = of_property_read_u32(np, "at25,byte-len", &val);
if (err) {
dev_dbg(&spi->dev, "invalid chip dt description\n");
goto fail;
}
chip.byte_len = val;
err = of_property_read_u32(np, "at25,addr-mode", &val);
if (err) {
dev_dbg(&spi->dev, "invalid chip dt description\n");
goto fail;
}
chip.flags = (u16)val;
err = of_property_read_u32(np, "at25,page-size", &val);
if (err) {
dev_dbg(&spi->dev, "invalid chip dt description\n");
goto fail;
}
chip.page_size = (u16)val;
} else {
dev_dbg(&spi->dev, "no chip description\n");
err = -ENODEV;
goto fail;
}
} else
chip = *(struct spi_eeprom *)spi->dev.platform_data;
/* For now we only support 8/16/24 bit addressing */ /* For now we only support 8/16/24 bit addressing */
if (chip->flags & EE_ADDR1) if (chip.flags & EE_ADDR1)
addrlen = 1; addrlen = 1;
else if (chip->flags & EE_ADDR2) else if (chip.flags & EE_ADDR2)
addrlen = 2; addrlen = 2;
else if (chip->flags & EE_ADDR3) else if (chip.flags & EE_ADDR3)
addrlen = 3; addrlen = 3;
else { else {
dev_dbg(&spi->dev, "unsupported address type\n"); dev_dbg(&spi->dev, "unsupported address type\n");
...@@ -348,7 +377,7 @@ static int at25_probe(struct spi_device *spi) ...@@ -348,7 +377,7 @@ static int at25_probe(struct spi_device *spi)
} }
mutex_init(&at25->lock); mutex_init(&at25->lock);
at25->chip = *chip; at25->chip = chip;
at25->spi = spi_dev_get(spi); at25->spi = spi_dev_get(spi);
dev_set_drvdata(&spi->dev, at25); dev_set_drvdata(&spi->dev, at25);
at25->addrlen = addrlen; at25->addrlen = addrlen;
...@@ -369,7 +398,7 @@ static int at25_probe(struct spi_device *spi) ...@@ -369,7 +398,7 @@ static int at25_probe(struct spi_device *spi)
at25->mem.read = at25_mem_read; at25->mem.read = at25_mem_read;
at25->bin.size = at25->chip.byte_len; at25->bin.size = at25->chip.byte_len;
if (!(chip->flags & EE_READONLY)) { if (!(chip.flags & EE_READONLY)) {
at25->bin.write = at25_bin_write; at25->bin.write = at25_bin_write;
at25->bin.attr.mode |= S_IWUSR; at25->bin.attr.mode |= S_IWUSR;
at25->mem.write = at25_mem_write; at25->mem.write = at25_mem_write;
...@@ -379,8 +408,8 @@ static int at25_probe(struct spi_device *spi) ...@@ -379,8 +408,8 @@ static int at25_probe(struct spi_device *spi)
if (err) if (err)
goto fail; goto fail;
if (chip->setup) if (chip.setup)
chip->setup(&at25->mem, chip->context); chip.setup(&at25->mem, chip.context);
dev_info(&spi->dev, "%Zd %s %s eeprom%s, pagesize %u\n", dev_info(&spi->dev, "%Zd %s %s eeprom%s, pagesize %u\n",
(at25->bin.size < 1024) (at25->bin.size < 1024)
...@@ -388,7 +417,7 @@ static int at25_probe(struct spi_device *spi) ...@@ -388,7 +417,7 @@ static int at25_probe(struct spi_device *spi)
: (at25->bin.size / 1024), : (at25->bin.size / 1024),
(at25->bin.size < 1024) ? "Byte" : "KByte", (at25->bin.size < 1024) ? "Byte" : "KByte",
at25->chip.name, at25->chip.name,
(chip->flags & EE_READONLY) ? " (readonly)" : "", (chip.flags & EE_READONLY) ? " (readonly)" : "",
at25->chip.page_size); at25->chip.page_size);
return 0; return 0;
fail: fail:
......
...@@ -30,6 +30,7 @@ ...@@ -30,6 +30,7 @@
static struct class *ilo_class; static struct class *ilo_class;
static unsigned int ilo_major; static unsigned int ilo_major;
static unsigned int max_ccb = MIN_CCB;
static char ilo_hwdev[MAX_ILO_DEV]; static char ilo_hwdev[MAX_ILO_DEV];
static inline int get_entry_id(int entry) static inline int get_entry_id(int entry)
...@@ -424,7 +425,7 @@ static void ilo_set_reset(struct ilo_hwinfo *hw) ...@@ -424,7 +425,7 @@ static void ilo_set_reset(struct ilo_hwinfo *hw)
* Mapped memory is zeroed on ilo reset, so set a per ccb flag * Mapped memory is zeroed on ilo reset, so set a per ccb flag
* to indicate that this ccb needs to be closed and reopened. * to indicate that this ccb needs to be closed and reopened.
*/ */
for (slot = 0; slot < MAX_CCB; slot++) { for (slot = 0; slot < max_ccb; slot++) {
if (!hw->ccb_alloc[slot]) if (!hw->ccb_alloc[slot])
continue; continue;
set_channel_reset(&hw->ccb_alloc[slot]->driver_ccb); set_channel_reset(&hw->ccb_alloc[slot]->driver_ccb);
...@@ -535,7 +536,7 @@ static int ilo_close(struct inode *ip, struct file *fp) ...@@ -535,7 +536,7 @@ static int ilo_close(struct inode *ip, struct file *fp)
struct ilo_hwinfo *hw; struct ilo_hwinfo *hw;
unsigned long flags; unsigned long flags;
slot = iminor(ip) % MAX_CCB; slot = iminor(ip) % max_ccb;
hw = container_of(ip->i_cdev, struct ilo_hwinfo, cdev); hw = container_of(ip->i_cdev, struct ilo_hwinfo, cdev);
spin_lock(&hw->open_lock); spin_lock(&hw->open_lock);
...@@ -566,7 +567,7 @@ static int ilo_open(struct inode *ip, struct file *fp) ...@@ -566,7 +567,7 @@ static int ilo_open(struct inode *ip, struct file *fp)
struct ilo_hwinfo *hw; struct ilo_hwinfo *hw;
unsigned long flags; unsigned long flags;
slot = iminor(ip) % MAX_CCB; slot = iminor(ip) % max_ccb;
hw = container_of(ip->i_cdev, struct ilo_hwinfo, cdev); hw = container_of(ip->i_cdev, struct ilo_hwinfo, cdev);
/* new ccb allocation */ /* new ccb allocation */
...@@ -663,7 +664,7 @@ static irqreturn_t ilo_isr(int irq, void *data) ...@@ -663,7 +664,7 @@ static irqreturn_t ilo_isr(int irq, void *data)
ilo_set_reset(hw); ilo_set_reset(hw);
} }
for (i = 0; i < MAX_CCB; i++) { for (i = 0; i < max_ccb; i++) {
if (!hw->ccb_alloc[i]) if (!hw->ccb_alloc[i])
continue; continue;
if (pending & (1 << i)) if (pending & (1 << i))
...@@ -697,14 +698,14 @@ static int __devinit ilo_map_device(struct pci_dev *pdev, struct ilo_hwinfo *hw) ...@@ -697,14 +698,14 @@ static int __devinit ilo_map_device(struct pci_dev *pdev, struct ilo_hwinfo *hw)
} }
/* map the adapter shared memory region */ /* map the adapter shared memory region */
hw->ram_vaddr = pci_iomap(pdev, 2, MAX_CCB * ILOHW_CCB_SZ); hw->ram_vaddr = pci_iomap(pdev, 2, max_ccb * ILOHW_CCB_SZ);
if (hw->ram_vaddr == NULL) { if (hw->ram_vaddr == NULL) {
dev_err(&pdev->dev, "Error mapping shared mem\n"); dev_err(&pdev->dev, "Error mapping shared mem\n");
goto mmio_free; goto mmio_free;
} }
/* map the doorbell aperture */ /* map the doorbell aperture */
hw->db_vaddr = pci_iomap(pdev, 3, MAX_CCB * ONE_DB_SIZE); hw->db_vaddr = pci_iomap(pdev, 3, max_ccb * ONE_DB_SIZE);
if (hw->db_vaddr == NULL) { if (hw->db_vaddr == NULL) {
dev_err(&pdev->dev, "Error mapping doorbell\n"); dev_err(&pdev->dev, "Error mapping doorbell\n");
goto ram_free; goto ram_free;
...@@ -727,7 +728,7 @@ static void ilo_remove(struct pci_dev *pdev) ...@@ -727,7 +728,7 @@ static void ilo_remove(struct pci_dev *pdev)
clear_device(ilo_hw); clear_device(ilo_hw);
minor = MINOR(ilo_hw->cdev.dev); minor = MINOR(ilo_hw->cdev.dev);
for (i = minor; i < minor + MAX_CCB; i++) for (i = minor; i < minor + max_ccb; i++)
device_destroy(ilo_class, MKDEV(ilo_major, i)); device_destroy(ilo_class, MKDEV(ilo_major, i));
cdev_del(&ilo_hw->cdev); cdev_del(&ilo_hw->cdev);
...@@ -737,7 +738,7 @@ static void ilo_remove(struct pci_dev *pdev) ...@@ -737,7 +738,7 @@ static void ilo_remove(struct pci_dev *pdev)
pci_release_regions(pdev); pci_release_regions(pdev);
pci_disable_device(pdev); pci_disable_device(pdev);
kfree(ilo_hw); kfree(ilo_hw);
ilo_hwdev[(minor / MAX_CCB)] = 0; ilo_hwdev[(minor / max_ccb)] = 0;
} }
static int __devinit ilo_probe(struct pci_dev *pdev, static int __devinit ilo_probe(struct pci_dev *pdev,
...@@ -746,6 +747,11 @@ static int __devinit ilo_probe(struct pci_dev *pdev, ...@@ -746,6 +747,11 @@ static int __devinit ilo_probe(struct pci_dev *pdev,
int devnum, minor, start, error; int devnum, minor, start, error;
struct ilo_hwinfo *ilo_hw; struct ilo_hwinfo *ilo_hw;
if (max_ccb > MAX_CCB)
max_ccb = MAX_CCB;
else if (max_ccb < MIN_CCB)
max_ccb = MIN_CCB;
/* find a free range for device files */ /* find a free range for device files */
for (devnum = 0; devnum < MAX_ILO_DEV; devnum++) { for (devnum = 0; devnum < MAX_ILO_DEV; devnum++) {
if (ilo_hwdev[devnum] == 0) { if (ilo_hwdev[devnum] == 0) {
...@@ -795,14 +801,14 @@ static int __devinit ilo_probe(struct pci_dev *pdev, ...@@ -795,14 +801,14 @@ static int __devinit ilo_probe(struct pci_dev *pdev,
cdev_init(&ilo_hw->cdev, &ilo_fops); cdev_init(&ilo_hw->cdev, &ilo_fops);
ilo_hw->cdev.owner = THIS_MODULE; ilo_hw->cdev.owner = THIS_MODULE;
start = devnum * MAX_CCB; start = devnum * max_ccb;
error = cdev_add(&ilo_hw->cdev, MKDEV(ilo_major, start), MAX_CCB); error = cdev_add(&ilo_hw->cdev, MKDEV(ilo_major, start), max_ccb);
if (error) { if (error) {
dev_err(&pdev->dev, "Could not add cdev\n"); dev_err(&pdev->dev, "Could not add cdev\n");
goto remove_isr; goto remove_isr;
} }
for (minor = 0 ; minor < MAX_CCB; minor++) { for (minor = 0 ; minor < max_ccb; minor++) {
struct device *dev; struct device *dev;
dev = device_create(ilo_class, &pdev->dev, dev = device_create(ilo_class, &pdev->dev,
MKDEV(ilo_major, minor), NULL, MKDEV(ilo_major, minor), NULL,
...@@ -879,11 +885,14 @@ static void __exit ilo_exit(void) ...@@ -879,11 +885,14 @@ static void __exit ilo_exit(void)
class_destroy(ilo_class); class_destroy(ilo_class);
} }
MODULE_VERSION("1.2"); MODULE_VERSION("1.3");
MODULE_ALIAS(ILO_NAME); MODULE_ALIAS(ILO_NAME);
MODULE_DESCRIPTION(ILO_NAME); MODULE_DESCRIPTION(ILO_NAME);
MODULE_AUTHOR("David Altobelli <david.altobelli@hp.com>"); MODULE_AUTHOR("David Altobelli <david.altobelli@hp.com>");
MODULE_LICENSE("GPL v2"); MODULE_LICENSE("GPL v2");
module_param(max_ccb, uint, 0444);
MODULE_PARM_DESC(max_ccb, "Maximum number of HP iLO channels to attach (8)");
module_init(ilo_init); module_init(ilo_init);
module_exit(ilo_exit); module_exit(ilo_exit);
...@@ -14,7 +14,9 @@ ...@@ -14,7 +14,9 @@
#define ILO_NAME "hpilo" #define ILO_NAME "hpilo"
/* max number of open channel control blocks per device, hw limited to 32 */ /* max number of open channel control blocks per device, hw limited to 32 */
#define MAX_CCB 8 #define MAX_CCB 24
/* min number of open channel control blocks per device, hw limited to 32 */
#define MIN_CCB 8
/* max number of supported devices */ /* max number of supported devices */
#define MAX_ILO_DEV 1 #define MAX_ILO_DEV 1
/* max number of files */ /* max number of files */
......
...@@ -162,6 +162,9 @@ int mei_hw_init(struct mei_device *dev) ...@@ -162,6 +162,9 @@ int mei_hw_init(struct mei_device *dev)
if ((dev->host_hw_state & H_IS) == H_IS) if ((dev->host_hw_state & H_IS) == H_IS)
mei_reg_write(dev, H_CSR, dev->host_hw_state); mei_reg_write(dev, H_CSR, dev->host_hw_state);
/* Doesn't change in runtime */
dev->hbuf_depth = (dev->host_hw_state & H_CBD) >> 24;
dev->recvd_msg = false; dev->recvd_msg = false;
dev_dbg(&dev->pdev->dev, "reset in start the mei device.\n"); dev_dbg(&dev->pdev->dev, "reset in start the mei device.\n");
...@@ -303,7 +306,6 @@ void mei_reset(struct mei_device *dev, int interrupts_enabled) ...@@ -303,7 +306,6 @@ void mei_reset(struct mei_device *dev, int interrupts_enabled)
dev->iamthif_cl.host_client_id); dev->iamthif_cl.host_client_id);
mei_reset_iamthif_params(dev); mei_reset_iamthif_params(dev);
dev->wd_due_counter = 0;
dev->extra_write_index = 0; dev->extra_write_index = 0;
} }
......
...@@ -58,16 +58,18 @@ void mei_disable_interrupts(struct mei_device *dev) ...@@ -58,16 +58,18 @@ void mei_disable_interrupts(struct mei_device *dev)
} }
/** /**
* _host_get_filled_slots - gets number of device filled buffer slots * mei_hbuf_filled_slots - gets number of device filled buffer slots
* *
* @device: the device structure * @device: the device structure
* *
* returns number of filled slots * returns number of filled slots
*/ */
static unsigned char _host_get_filled_slots(const struct mei_device *dev) static unsigned char mei_hbuf_filled_slots(struct mei_device *dev)
{ {
char read_ptr, write_ptr; char read_ptr, write_ptr;
dev->host_hw_state = mei_hcsr_read(dev);
read_ptr = (char) ((dev->host_hw_state & H_CBRP) >> 8); read_ptr = (char) ((dev->host_hw_state & H_CBRP) >> 8);
write_ptr = (char) ((dev->host_hw_state & H_CBWP) >> 16); write_ptr = (char) ((dev->host_hw_state & H_CBWP) >> 16);
...@@ -75,43 +77,33 @@ static unsigned char _host_get_filled_slots(const struct mei_device *dev) ...@@ -75,43 +77,33 @@ static unsigned char _host_get_filled_slots(const struct mei_device *dev)
} }
/** /**
* mei_host_buffer_is_empty - checks if host buffer is empty. * mei_hbuf_is_empty - checks if host buffer is empty.
* *
* @dev: the device structure * @dev: the device structure
* *
* returns 1 if empty, 0 - otherwise. * returns true if empty, false - otherwise.
*/ */
int mei_host_buffer_is_empty(struct mei_device *dev) bool mei_hbuf_is_empty(struct mei_device *dev)
{ {
unsigned char filled_slots; return mei_hbuf_filled_slots(dev) == 0;
dev->host_hw_state = mei_hcsr_read(dev);
filled_slots = _host_get_filled_slots(dev);
if (filled_slots == 0)
return 1;
return 0;
} }
/** /**
* mei_count_empty_write_slots - counts write empty slots. * mei_hbuf_empty_slots - counts write empty slots.
* *
* @dev: the device structure * @dev: the device structure
* *
* returns -1(ESLOTS_OVERFLOW) if overflow, otherwise empty slots count * returns -1(ESLOTS_OVERFLOW) if overflow, otherwise empty slots count
*/ */
int mei_count_empty_write_slots(struct mei_device *dev) int mei_hbuf_empty_slots(struct mei_device *dev)
{ {
unsigned char buffer_depth, filled_slots, empty_slots; unsigned char filled_slots, empty_slots;
dev->host_hw_state = mei_hcsr_read(dev); filled_slots = mei_hbuf_filled_slots(dev);
buffer_depth = (unsigned char) ((dev->host_hw_state & H_CBD) >> 24); empty_slots = dev->hbuf_depth - filled_slots;
filled_slots = _host_get_filled_slots(dev);
empty_slots = buffer_depth - filled_slots;
/* check for overflow */ /* check for overflow */
if (filled_slots > buffer_depth) if (filled_slots > dev->hbuf_depth)
return -EOVERFLOW; return -EOVERFLOW;
return empty_slots; return empty_slots;
...@@ -127,52 +119,39 @@ int mei_count_empty_write_slots(struct mei_device *dev) ...@@ -127,52 +119,39 @@ int mei_count_empty_write_slots(struct mei_device *dev)
* *
* This function returns -EIO if write has failed * This function returns -EIO if write has failed
*/ */
int mei_write_message(struct mei_device *dev, int mei_write_message(struct mei_device *dev, struct mei_msg_hdr *header,
struct mei_msg_hdr *header, unsigned char *buf, unsigned long length)
unsigned char *write_buffer,
unsigned long write_length)
{ {
u32 temp_msg = 0; unsigned long rem, dw_cnt;
unsigned long bytes_written = 0; u32 *reg_buf = (u32 *)buf;
unsigned char buffer_depth, filled_slots, empty_slots; int i;
unsigned long dw_to_write; int empty_slots;
dev->host_hw_state = mei_hcsr_read(dev);
dev_dbg(&dev->pdev->dev,
"host_hw_state = 0x%08x.\n",
dev->host_hw_state);
dev_dbg(&dev->pdev->dev, dev_dbg(&dev->pdev->dev,
"mei_write_message header=%08x.\n", "mei_write_message header=%08x.\n",
*((u32 *) header)); *((u32 *) header));
buffer_depth = (unsigned char) ((dev->host_hw_state & H_CBD) >> 24); empty_slots = mei_hbuf_empty_slots(dev);
filled_slots = _host_get_filled_slots(dev); dev_dbg(&dev->pdev->dev, "empty slots = %hu.\n", empty_slots);
empty_slots = buffer_depth - filled_slots;
dev_dbg(&dev->pdev->dev,
"filled = %hu, empty = %hu.\n",
filled_slots, empty_slots);
dw_to_write = ((write_length + 3) / 4);
if (dw_to_write > empty_slots) dw_cnt = mei_data2slots(length);
if (empty_slots < 0 || dw_cnt > empty_slots)
return -EIO; return -EIO;
mei_reg_write(dev, H_CB_WW, *((u32 *) header)); mei_reg_write(dev, H_CB_WW, *((u32 *) header));
while (write_length >= 4) { for (i = 0; i < length / 4; i++)
mei_reg_write(dev, H_CB_WW, mei_reg_write(dev, H_CB_WW, reg_buf[i]);
*(u32 *) (write_buffer + bytes_written));
bytes_written += 4;
write_length -= 4;
}
if (write_length > 0) { rem = length & 0x3;
memcpy(&temp_msg, &write_buffer[bytes_written], write_length); if (rem > 0) {
mei_reg_write(dev, H_CB_WW, temp_msg); u32 reg = 0;
memcpy(&reg, &buf[length - rem], rem);
mei_reg_write(dev, H_CB_WW, reg);
} }
dev->host_hw_state = mei_hcsr_read(dev);
dev->host_hw_state |= H_IG; dev->host_hw_state |= H_IG;
mei_hcsr_set(dev); mei_hcsr_set(dev);
dev->me_hw_state = mei_mecsr_read(dev); dev->me_hw_state = mei_mecsr_read(dev);
......
...@@ -41,14 +41,28 @@ int mei_write_message(struct mei_device *dev, ...@@ -41,14 +41,28 @@ int mei_write_message(struct mei_device *dev,
unsigned char *write_buffer, unsigned char *write_buffer,
unsigned long write_length); unsigned long write_length);
int mei_host_buffer_is_empty(struct mei_device *dev); bool mei_hbuf_is_empty(struct mei_device *dev);
int mei_hbuf_empty_slots(struct mei_device *dev);
static inline size_t mei_hbuf_max_data(const struct mei_device *dev)
{
return dev->hbuf_depth * sizeof(u32) - sizeof(struct mei_msg_hdr);
}
/* get slots (dwords) from a message length + header (bytes) */
static inline unsigned char mei_data2slots(size_t length)
{
return DIV_ROUND_UP(sizeof(struct mei_msg_hdr) + length, 4);
}
int mei_count_full_read_slots(struct mei_device *dev); int mei_count_full_read_slots(struct mei_device *dev);
int mei_count_empty_write_slots(struct mei_device *dev);
int mei_flow_ctrl_creds(struct mei_device *dev, struct mei_cl *cl); int mei_flow_ctrl_creds(struct mei_device *dev, struct mei_cl *cl);
int mei_wd_send(struct mei_device *dev); int mei_wd_send(struct mei_device *dev);
int mei_wd_stop(struct mei_device *dev, bool preserve); int mei_wd_stop(struct mei_device *dev, bool preserve);
int mei_wd_host_init(struct mei_device *dev); int mei_wd_host_init(struct mei_device *dev);
......
This diff is collapsed.
...@@ -481,12 +481,8 @@ int amthi_write(struct mei_device *dev, struct mei_cl_cb *cb) ...@@ -481,12 +481,8 @@ int amthi_write(struct mei_device *dev, struct mei_cl_cb *cb)
if (ret && dev->mei_host_buffer_is_empty) { if (ret && dev->mei_host_buffer_is_empty) {
ret = 0; ret = 0;
dev->mei_host_buffer_is_empty = false; dev->mei_host_buffer_is_empty = false;
if (cb->request_buffer.size > if (cb->request_buffer.size > mei_hbuf_max_data(dev)) {
(((dev->host_hw_state & H_CBD) >> 24) * sizeof(u32)) mei_hdr.length = mei_hbuf_max_data(dev);
-sizeof(struct mei_msg_hdr)) {
mei_hdr.length =
(((dev->host_hw_state & H_CBD) >> 24) *
sizeof(u32)) - sizeof(struct mei_msg_hdr);
mei_hdr.msg_complete = 0; mei_hdr.msg_complete = 0;
} else { } else {
mei_hdr.length = cb->request_buffer.size; mei_hdr.length = cb->request_buffer.size;
......
...@@ -714,13 +714,8 @@ static ssize_t mei_write(struct file *file, const char __user *ubuf, ...@@ -714,13 +714,8 @@ static ssize_t mei_write(struct file *file, const char __user *ubuf,
if (rets && dev->mei_host_buffer_is_empty) { if (rets && dev->mei_host_buffer_is_empty) {
rets = 0; rets = 0;
dev->mei_host_buffer_is_empty = false; dev->mei_host_buffer_is_empty = false;
if (length > ((((dev->host_hw_state & H_CBD) >> 24) * if (length > mei_hbuf_max_data(dev)) {
sizeof(u32)) - sizeof(struct mei_msg_hdr))) { mei_hdr.length = mei_hbuf_max_data(dev);
mei_hdr.length =
(((dev->host_hw_state & H_CBD) >> 24) *
sizeof(u32)) -
sizeof(struct mei_msg_hdr);
mei_hdr.msg_complete = 0; mei_hdr.msg_complete = 0;
} else { } else {
mei_hdr.length = length; mei_hdr.length = length;
...@@ -1187,44 +1182,7 @@ static struct pci_driver mei_driver = { ...@@ -1187,44 +1182,7 @@ static struct pci_driver mei_driver = {
.driver.pm = MEI_PM_OPS, .driver.pm = MEI_PM_OPS,
}; };
/** module_pci_driver(mei_driver);
* mei_init_module - Driver Registration Routine
*
* mei_init_module is the first routine called when the driver is
* loaded. All it does is to register with the PCI subsystem.
*
* returns 0 on success, <0 on failure.
*/
static int __init mei_init_module(void)
{
int ret;
pr_debug("loading.\n");
/* init pci module */
ret = pci_register_driver(&mei_driver);
if (ret < 0)
pr_err("error registering driver.\n");
return ret;
}
module_init(mei_init_module);
/**
* mei_exit_module - Driver Exit Cleanup Routine
*
* mei_exit_module is called just before the driver is removed
* from memory.
*/
static void __exit mei_exit_module(void)
{
pci_unregister_driver(&mei_driver);
pr_debug("unloaded successfully.\n");
}
module_exit(mei_exit_module);
MODULE_AUTHOR("Intel Corporation"); MODULE_AUTHOR("Intel Corporation");
MODULE_DESCRIPTION("Intel(R) Management Engine Interface"); MODULE_DESCRIPTION("Intel(R) Management Engine Interface");
......
...@@ -167,7 +167,10 @@ struct mei_io_list { ...@@ -167,7 +167,10 @@ struct mei_io_list {
struct mei_cl_cb mei_cb; struct mei_cl_cb mei_cb;
}; };
/* MEI private device struct */ /**
* struct mei_deive - MEI private device struct
* @hbuf_depth - depth of host(write) buffer
*/
struct mei_device { struct mei_device {
struct pci_dev *pdev; /* pointer to pci device struct */ struct pci_dev *pdev; /* pointer to pci device struct */
/* /*
...@@ -205,6 +208,7 @@ struct mei_device { ...@@ -205,6 +208,7 @@ struct mei_device {
*/ */
u32 host_hw_state; u32 host_hw_state;
u32 me_hw_state; u32 me_hw_state;
u8 hbuf_depth;
/* /*
* waiting queue for receive message from FW * waiting queue for receive message from FW
*/ */
...@@ -237,15 +241,14 @@ struct mei_device { ...@@ -237,15 +241,14 @@ struct mei_device {
bool mei_host_buffer_is_empty; bool mei_host_buffer_is_empty;
struct mei_cl wd_cl; struct mei_cl wd_cl;
bool wd_interface_reg;
bool wd_pending; bool wd_pending;
bool wd_stopped; bool wd_stopped;
bool wd_bypass; /* if false, don't refresh watchdog ME client */ bool wd_bypass; /* if false, don't refresh watchdog ME client */
u16 wd_timeout; /* seconds ((wd_data[1] << 8) + wd_data[0]) */ u16 wd_timeout; /* seconds ((wd_data[1] << 8) + wd_data[0]) */
u16 wd_due_counter;
unsigned char wd_data[MEI_START_WD_DATA_SIZE]; unsigned char wd_data[MEI_START_WD_DATA_SIZE];
struct file *iamthif_file_object; struct file *iamthif_file_object;
struct mei_cl iamthif_cl; struct mei_cl iamthif_cl;
struct mei_cl_cb *iamthif_current_cb; struct mei_cl_cb *iamthif_current_cb;
...@@ -259,8 +262,6 @@ struct mei_device { ...@@ -259,8 +262,6 @@ struct mei_device {
bool iamthif_flow_control_pending; bool iamthif_flow_control_pending;
bool iamthif_ioctl; bool iamthif_ioctl;
bool iamthif_canceled; bool iamthif_canceled;
bool wd_interface_reg;
}; };
...@@ -361,7 +362,8 @@ int mei_find_me_client_index(const struct mei_device *dev, uuid_le cuuid); ...@@ -361,7 +362,8 @@ int mei_find_me_client_index(const struct mei_device *dev, uuid_le cuuid);
* *
* returns register value (u32) * returns register value (u32)
*/ */
static inline u32 mei_reg_read(struct mei_device *dev, unsigned long offset) static inline u32 mei_reg_read(const struct mei_device *dev,
unsigned long offset)
{ {
return ioread32(dev->mem_addr + offset); return ioread32(dev->mem_addr + offset);
} }
...@@ -373,8 +375,8 @@ static inline u32 mei_reg_read(struct mei_device *dev, unsigned long offset) ...@@ -373,8 +375,8 @@ static inline u32 mei_reg_read(struct mei_device *dev, unsigned long offset)
* @offset: offset from which to write the data * @offset: offset from which to write the data
* @value: register value to write (u32) * @value: register value to write (u32)
*/ */
static inline void mei_reg_write(struct mei_device *dev, static inline void mei_reg_write(const struct mei_device *dev,
unsigned long offset, u32 value) unsigned long offset, u32 value)
{ {
iowrite32(value, dev->mem_addr + offset); iowrite32(value, dev->mem_addr + offset);
} }
...@@ -386,7 +388,7 @@ static inline void mei_reg_write(struct mei_device *dev, ...@@ -386,7 +388,7 @@ static inline void mei_reg_write(struct mei_device *dev,
* *
* returns the byte read. * returns the byte read.
*/ */
static inline u32 mei_hcsr_read(struct mei_device *dev) static inline u32 mei_hcsr_read(const struct mei_device *dev)
{ {
return mei_reg_read(dev, H_CSR); return mei_reg_read(dev, H_CSR);
} }
...@@ -398,7 +400,7 @@ static inline u32 mei_hcsr_read(struct mei_device *dev) ...@@ -398,7 +400,7 @@ static inline u32 mei_hcsr_read(struct mei_device *dev)
* *
* returns ME_CSR_HA register value (u32) * returns ME_CSR_HA register value (u32)
*/ */
static inline u32 mei_mecsr_read(struct mei_device *dev) static inline u32 mei_mecsr_read(const struct mei_device *dev)
{ {
return mei_reg_read(dev, ME_CSR_HA); return mei_reg_read(dev, ME_CSR_HA);
} }
...@@ -410,7 +412,7 @@ static inline u32 mei_mecsr_read(struct mei_device *dev) ...@@ -410,7 +412,7 @@ static inline u32 mei_mecsr_read(struct mei_device *dev)
* *
* returns ME_CB_RW register value (u32) * returns ME_CB_RW register value (u32)
*/ */
static inline u32 mei_mecbrw_read(struct mei_device *dev) static inline u32 mei_mecbrw_read(const struct mei_device *dev)
{ {
return mei_reg_read(dev, ME_CB_RW); return mei_reg_read(dev, ME_CB_RW);
} }
......
...@@ -53,11 +53,12 @@ static void mei_wd_set_start_timeout(struct mei_device *dev, u16 timeout) ...@@ -53,11 +53,12 @@ static void mei_wd_set_start_timeout(struct mei_device *dev, u16 timeout)
} }
/** /**
* host_init_wd - mei initialization wd. * mei_wd_host_init - connect to the watchdog client
* *
* @dev: the device structure * @dev: the device structure
* returns -ENENT if wd client cannot be found * returns -ENENT if wd client cannot be found
* -EIO if write has failed * -EIO if write has failed
* 0 on success
*/ */
int mei_wd_host_init(struct mei_device *dev) int mei_wd_host_init(struct mei_device *dev)
{ {
...@@ -137,7 +138,6 @@ int mei_wd_stop(struct mei_device *dev, bool preserve) ...@@ -137,7 +138,6 @@ int mei_wd_stop(struct mei_device *dev, bool preserve)
return 0; return 0;
dev->wd_timeout = 0; dev->wd_timeout = 0;
dev->wd_due_counter = 0;
memcpy(dev->wd_data, mei_stop_wd_params, MEI_WD_PARAMS_SIZE); memcpy(dev->wd_data, mei_stop_wd_params, MEI_WD_PARAMS_SIZE);
dev->stop = true; dev->stop = true;
...@@ -357,8 +357,6 @@ void mei_watchdog_register(struct mei_device *dev) ...@@ -357,8 +357,6 @@ void mei_watchdog_register(struct mei_device *dev)
{ {
dev_dbg(&dev->pdev->dev, "dev->wd_timeout =%d.\n", dev->wd_timeout); dev_dbg(&dev->pdev->dev, "dev->wd_timeout =%d.\n", dev->wd_timeout);
dev->wd_due_counter = !!dev->wd_timeout;
if (watchdog_register_device(&amt_wd_dev)) { if (watchdog_register_device(&amt_wd_dev)) {
dev_err(&dev->pdev->dev, dev_err(&dev->pdev->dev,
"wd: unable to register watchdog device.\n"); "wd: unable to register watchdog device.\n");
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment