Commit 375d65db authored by Alan Cox's avatar Alan Cox Committed by Greg Kroah-Hartman

Staging: rar and memrar updates

rar: perform a clean up pass

- Move to a registration model where each RAR is claimed/unclaimed
- Use that to fix the client stuff (one client per RAR so no need to queue stuff)
- Support unregister so drivers can rmmod themselves safely
- Fix locking hang on calling rar lock from rar callback
- Clean up
- Kerneldoc

Folded in the memrar update as Greg asked

- Fix various unload related bugs
- Use the per RAR allocator/deallocator
- Add kerneldoc
Signed-off-by: default avatarAlan Cox <alan@linux.intel.com>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@suse.de>
parent a9d26f00
...@@ -114,6 +114,7 @@ struct memrar_rar_info { ...@@ -114,6 +114,7 @@ struct memrar_rar_info {
struct memrar_allocator *allocator; struct memrar_allocator *allocator;
struct memrar_buffer_info buffers; struct memrar_buffer_info buffers;
struct mutex lock; struct mutex lock;
int allocated; /* True if we own this RAR */
}; };
/* /*
...@@ -150,11 +151,13 @@ static struct memrar_rar_info *memrar_get_rar_info(u32 vaddr) ...@@ -150,11 +151,13 @@ static struct memrar_rar_info *memrar_get_rar_info(u32 vaddr)
return NULL; return NULL;
} }
/* /**
* Retrieve bus address from given handle. * memrar_get_bus address - handle to bus address
*
* Retrieve bus address from given handle.
* *
* Returns address corresponding to given handle. Zero if handle is * Returns address corresponding to given handle. Zero if handle is
* invalid. * invalid.
*/ */
static dma_addr_t memrar_get_bus_address( static dma_addr_t memrar_get_bus_address(
struct memrar_rar_info *rar, struct memrar_rar_info *rar,
...@@ -176,11 +179,13 @@ static dma_addr_t memrar_get_bus_address( ...@@ -176,11 +179,13 @@ static dma_addr_t memrar_get_bus_address(
return rar->base + (vaddr - iobase); return rar->base + (vaddr - iobase);
} }
/* /**
* Retrieve physical address from given handle. * memrar_get_physical_address - handle to physical address
*
* Retrieve physical address from given handle.
* *
* Returns address corresponding to given handle. Zero if handle is * Returns address corresponding to given handle. Zero if handle is
* invalid. * invalid.
*/ */
static dma_addr_t memrar_get_physical_address( static dma_addr_t memrar_get_physical_address(
struct memrar_rar_info *rar, struct memrar_rar_info *rar,
...@@ -195,11 +200,15 @@ static dma_addr_t memrar_get_physical_address( ...@@ -195,11 +200,15 @@ static dma_addr_t memrar_get_physical_address(
return memrar_get_bus_address(rar, vaddr); return memrar_get_bus_address(rar, vaddr);
} }
/* /**
* Core block release code. * memrar_release_block - release a block to the pool
* @kref: kref of block
*
* Core block release code. A node has hit zero references so can
* be released and the lists must be updated.
* *
* Note: This code removes the node from a list. Make sure any list * Note: This code removes the node from a list. Make sure any list
* iteration is performed using list_for_each_safe(). * iteration is performed using list_for_each_safe().
*/ */
static void memrar_release_block_i(struct kref *ref) static void memrar_release_block_i(struct kref *ref)
{ {
...@@ -224,10 +233,15 @@ static void memrar_release_block_i(struct kref *ref) ...@@ -224,10 +233,15 @@ static void memrar_release_block_i(struct kref *ref)
kfree(node); kfree(node);
} }
/* /**
* Initialize RAR parameters, such as bus addresses, etc. * memrar_init_rar_resources - configure a RAR
* @rarnum: rar that has been allocated
* @devname: name of our device
*
* Initialize RAR parameters, such as bus addresses, etc and make
* the resource accessible.
*/ */
static int memrar_init_rar_resources(char const *devname) static int memrar_init_rar_resources(int rarnum, char const *devname)
{ {
/* ---- Sanity Checks ---- /* ---- Sanity Checks ----
* 1. RAR bus addresses in both Lincroft and Langwell RAR * 1. RAR bus addresses in both Lincroft and Langwell RAR
...@@ -258,162 +272,95 @@ static int memrar_init_rar_resources(char const *devname) ...@@ -258,162 +272,95 @@ static int memrar_init_rar_resources(char const *devname)
*/ */
static size_t const RAR_BLOCK_SIZE = PAGE_SIZE; static size_t const RAR_BLOCK_SIZE = PAGE_SIZE;
int z; dma_addr_t low, high;
int found_rar = 0; struct memrar_rar_info * const rar = &memrars[rarnum];
BUG_ON(MRST_NUM_RAR != ARRAY_SIZE(memrars)); BUG_ON(MRST_NUM_RAR != ARRAY_SIZE(memrars));
BUG_ON(!memrar_is_valid_rar_type(rarnum));
BUG_ON(rar->allocated);
for (z = 0; z != MRST_NUM_RAR; ++z) { mutex_init(&rar->lock);
dma_addr_t low, high;
struct memrar_rar_info * const rar = &memrars[z];
BUG_ON(!memrar_is_valid_rar_type(z));
mutex_init(&rar->lock);
/*
* Initialize the process table before we reach any
* code that exit on failure since the finalization
* code requires an initialized list.
*/
INIT_LIST_HEAD(&rar->buffers.list);
if (rar_get_address(z, &low, &high) != 0) {
/* No RAR is available. */
break;
} else if (low == 0 || high == 0) {
/*
* We don't immediately break out of the loop
* since the next type of RAR may be enabled.
*/
rar->base = 0;
rar->length = 0;
rar->iobase = NULL;
rar->allocator = NULL;
continue;
}
/*
* @todo Verify that LNC and LNW RAR register contents
* addresses, security, etc are compatible and
* consistent).
*/
rar->length = high - low + 1;
/* Claim RAR memory as our own. */
if (request_mem_region(low, rar->length, devname) == NULL) {
rar->length = 0;
pr_err("%s: Unable to claim RAR[%d] memory.\n",
devname,
z);
pr_err("%s: RAR[%d] disabled.\n", devname, z);
/*
* Rather than break out of the loop by
* returning -EBUSY, for example, we may be
* able to claim memory of the next RAR region
* as our own.
*/
continue;
}
rar->base = low;
/*
* Now map it into the kernel address space.
*
* Note that the RAR memory may only be accessed by IA
* when debugging. Otherwise attempts to access the
* RAR memory when it is locked down will result in
* behavior similar to writing to /dev/null and
* reading from /dev/zero. This behavior is enforced
* by the hardware. Even if we don't access the
* memory, mapping it into the kernel provides us with
* a convenient RAR handle to bus address mapping.
*/
rar->iobase = ioremap_nocache(rar->base, rar->length);
if (rar->iobase == NULL) {
pr_err("%s: Unable to map RAR memory.\n",
devname);
return -ENOMEM;
}
/* Initialize corresponding memory allocator. */
rar->allocator = memrar_create_allocator(
(unsigned long) rar->iobase,
rar->length,
RAR_BLOCK_SIZE);
if (rar->allocator == NULL)
return -1;
/* /*
* ------------------------------------------------- * Initialize the process table before we reach any
* Make sure all RARs handled by us are locked down. * code that exit on failure since the finalization
* ------------------------------------------------- * code requires an initialized list.
*/ */
INIT_LIST_HEAD(&rar->buffers.list);
/* Enable RAR protection on the Lincroft side. */ if (rar_get_address(rarnum, &low, &high) != 0)
if (0) { /* No RAR is available. */
/* return -ENODEV;
* This is mostly a sanity check since the
* vendor should have locked down RAR in the if (low == 0 || high == 0) {
* SMIP header RAR configuration. rar->base = 0;
*/ rar->length = 0;
rar_lock(z); rar->iobase = NULL;
} else { rar->allocator = NULL;
pr_warning("%s: LNC RAR[%d] no lock sanity check.\n", return -ENOSPC;
devname, }
z);
}
/* ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ */ /*
/* |||||||||||||||||||||||||||||||||||||||||||||||||| */ * @todo Verify that LNC and LNW RAR register contents
* addresses, security, etc are compatible and
* consistent).
*/
/* rar->length = high - low + 1;
* It would be nice if we could verify that RAR
* protection on the Langwell side is enabled, but
* there is no way to do that from here. The
* necessary Langwell RAR registers are not accessible
* from the Lincroft (IA) side.
*
* Hopefully the ODM did the right thing and enabled
* Langwell side RAR protection in the integrated
* firmware SMIP header.
*/
pr_info("%s: BRAR[%d] bus address range = " /* Claim RAR memory as our own. */
"[0x%lx, 0x%lx]\n", if (request_mem_region(low, rar->length, devname) == NULL) {
devname, rar->length = 0;
z, pr_err("%s: Unable to claim RAR[%d] memory.\n", devname, rarnum);
(unsigned long) low, pr_err("%s: RAR[%d] disabled.\n", devname, rarnum);
(unsigned long) high); return -EBUSY;
}
pr_info("%s: BRAR[%d] size = %zu KiB\n", rar->base = low;
devname,
z,
rar->allocator->capacity / 1024);
found_rar = 1; /*
* Now map it into the kernel address space.
*
* Note that the RAR memory may only be accessed by IA
* when debugging. Otherwise attempts to access the
* RAR memory when it is locked down will result in
* behavior similar to writing to /dev/null and
* reading from /dev/zero. This behavior is enforced
* by the hardware. Even if we don't access the
* memory, mapping it into the kernel provides us with
* a convenient RAR handle to bus address mapping.
*/
rar->iobase = ioremap_nocache(rar->base, rar->length);
if (rar->iobase == NULL) {
pr_err("%s: Unable to map RAR memory.\n", devname);
release_mem_region(low, rar->length);
return -ENOMEM;
} }
if (!found_rar) { /* Initialize corresponding memory allocator. */
/* rar->allocator = memrar_create_allocator((unsigned long) rar->iobase,
* No RAR support. Don't bother continuing. rar->length, RAR_BLOCK_SIZE);
* if (rar->allocator == NULL) {
* Note that this is not a failure. iounmap(rar->iobase);
*/ release_mem_region(low, rar->length);
pr_info("%s: No Moorestown RAR support available.\n", return -ENOMEM;
devname);
return -ENODEV;
} }
pr_info("%s: BRAR[%d] bus address range = [0x%lx, 0x%lx]\n",
devname, rarnum, (unsigned long) low, (unsigned long) high);
pr_info("%s: BRAR[%d] size = %zu KiB\n",
devname, rarnum, rar->allocator->capacity / 1024);
rar->allocated = 1;
return 0; return 0;
} }
/* /**
* Finalize RAR resources. * memrar_fini_rar_resources - free up RAR resources
*
* Finalize RAR resources. Free up the resource tables, hand the memory
* back to the kernel, unmap the device and release the address space.
*/ */
static void memrar_fini_rar_resources(void) static void memrar_fini_rar_resources(void)
{ {
...@@ -429,6 +376,9 @@ static void memrar_fini_rar_resources(void) ...@@ -429,6 +376,9 @@ static void memrar_fini_rar_resources(void)
for (z = MRST_NUM_RAR; z-- != 0; ) { for (z = MRST_NUM_RAR; z-- != 0; ) {
struct memrar_rar_info * const rar = &memrars[z]; struct memrar_rar_info * const rar = &memrars[z];
if (!rar->allocated)
continue;
/* Clean up remaining resources. */ /* Clean up remaining resources. */
list_for_each_entry_safe(pos, list_for_each_entry_safe(pos,
...@@ -442,15 +392,25 @@ static void memrar_fini_rar_resources(void) ...@@ -442,15 +392,25 @@ static void memrar_fini_rar_resources(void)
rar->allocator = NULL; rar->allocator = NULL;
iounmap(rar->iobase); iounmap(rar->iobase);
rar->iobase = NULL;
release_mem_region(rar->base, rar->length); release_mem_region(rar->base, rar->length);
rar->base = 0;
rar->iobase = NULL;
rar->base = 0;
rar->length = 0; rar->length = 0;
unregister_rar(z);
} }
} }
/**
* memrar_reserve_block - handle an allocation request
* @request: block being requested
* @filp: owner it is tied to
*
* Allocate a block of the requested RAR. If successful return the
* request object filled in and zero, if not report an error code
*/
static long memrar_reserve_block(struct RAR_buffer *request, static long memrar_reserve_block(struct RAR_buffer *request,
struct file *filp) struct file *filp)
{ {
...@@ -465,6 +425,8 @@ static long memrar_reserve_block(struct RAR_buffer *request, ...@@ -465,6 +425,8 @@ static long memrar_reserve_block(struct RAR_buffer *request,
return -EINVAL; return -EINVAL;
rar = &memrars[rinfo->type]; rar = &memrars[rinfo->type];
if (!rar->allocated)
return -ENODEV;
/* Reserve memory in RAR. */ /* Reserve memory in RAR. */
handle = memrar_allocator_alloc(rar->allocator, rinfo->size); handle = memrar_allocator_alloc(rar->allocator, rinfo->size);
...@@ -504,6 +466,14 @@ static long memrar_reserve_block(struct RAR_buffer *request, ...@@ -504,6 +466,14 @@ static long memrar_reserve_block(struct RAR_buffer *request,
return 0; return 0;
} }
/**
* memrar_release_block - release a RAR block
* @addr: address in RAR space
*
* Release a previously allocated block. Releases act on complete
* blocks, partially freeing a block is not supported
*/
static long memrar_release_block(u32 addr) static long memrar_release_block(u32 addr)
{ {
struct memrar_buffer_info *pos; struct memrar_buffer_info *pos;
...@@ -512,7 +482,7 @@ static long memrar_release_block(u32 addr) ...@@ -512,7 +482,7 @@ static long memrar_release_block(u32 addr)
long result = -EINVAL; long result = -EINVAL;
if (rar == NULL) if (rar == NULL)
return -EFAULT; return -ENOENT;
mutex_lock(&rar->lock); mutex_lock(&rar->lock);
...@@ -550,34 +520,48 @@ static long memrar_release_block(u32 addr) ...@@ -550,34 +520,48 @@ static long memrar_release_block(u32 addr)
return result; return result;
} }
/**
* memrar_get_stats - read statistics for a RAR
* @r: statistics to be filled in
*
* Returns the statistics data for the RAR, or an error code if
* the request cannot be completed
*/
static long memrar_get_stat(struct RAR_stat *r) static long memrar_get_stat(struct RAR_stat *r)
{ {
long result = -EINVAL; struct memrar_allocator *allocator;
if (likely(r != NULL) && memrar_is_valid_rar_type(r->type)) {
struct memrar_allocator * const allocator =
memrars[r->type].allocator;
BUG_ON(allocator == NULL);
/* if (!memrar_is_valid_rar_type(r->type))
* Allocator capacity doesn't change over time. No return -EINVAL;
* need to synchronize.
*/
r->capacity = allocator->capacity;
mutex_lock(&allocator->lock); if (!memrars[r->type].allocated)
return -ENODEV;
r->largest_block_size = allocator->largest_free_area; allocator = memrars[r->type].allocator;
mutex_unlock(&allocator->lock); BUG_ON(allocator == NULL);
result = 0; /*
} * Allocator capacity doesn't change over time. No
* need to synchronize.
*/
r->capacity = allocator->capacity;
return result; mutex_lock(&allocator->lock);
r->largest_block_size = allocator->largest_free_area;
mutex_unlock(&allocator->lock);
return 0;
} }
/**
* memrar_ioctl - ioctl callback
* @filp: file issuing the request
* @cmd: command
* @arg: pointer to control information
*
* Perform one of the ioctls supported by the memrar device
*/
static long memrar_ioctl(struct file *filp, static long memrar_ioctl(struct file *filp,
unsigned int cmd, unsigned int cmd,
unsigned long arg) unsigned long arg)
...@@ -640,6 +624,15 @@ static long memrar_ioctl(struct file *filp, ...@@ -640,6 +624,15 @@ static long memrar_ioctl(struct file *filp,
return 0; return 0;
} }
/**
* memrar_mmap - mmap helper for deubgging
* @filp: handle doing the mapping
* @vma: memory area
*
* Support the mmap operation on the RAR space for debugging systems
* when the memory is not locked down.
*/
static int memrar_mmap(struct file *filp, struct vm_area_struct *vma) static int memrar_mmap(struct file *filp, struct vm_area_struct *vma)
{ {
/* /*
...@@ -660,9 +653,12 @@ static int memrar_mmap(struct file *filp, struct vm_area_struct *vma) ...@@ -660,9 +653,12 @@ static int memrar_mmap(struct file *filp, struct vm_area_struct *vma)
unsigned long const handle = vma->vm_pgoff << PAGE_SHIFT; unsigned long const handle = vma->vm_pgoff << PAGE_SHIFT;
struct memrar_rar_info * const rar = memrar_get_rar_info(handle); struct memrar_rar_info * const rar = memrar_get_rar_info(handle);
unsigned long pfn; unsigned long pfn;
/* Only allow priviledged apps to go poking around this way */
if (!capable(CAP_SYS_RAWIO))
return -EPERM;
/* Invalid RAR handle or size passed to mmap(). */ /* Invalid RAR handle or size passed to mmap(). */
if (rar == NULL if (rar == NULL
|| handle == 0 || handle == 0
...@@ -698,13 +694,32 @@ static int memrar_mmap(struct file *filp, struct vm_area_struct *vma) ...@@ -698,13 +694,32 @@ static int memrar_mmap(struct file *filp, struct vm_area_struct *vma)
return 0; return 0;
} }
/**
* memrar_open - device open method
* @inode: inode to open
* @filp: file handle
*
* As we support multiple arbitary opens there is no work to be done
* really.
*/
static int memrar_open(struct inode *inode, struct file *filp) static int memrar_open(struct inode *inode, struct file *filp)
{ {
/* Nothing to do yet. */ nonseekable_open(inode, filp);
return 0; return 0;
} }
/**
* memrar_release - close method for miscev
* @inode: inode of device
* @filp: handle that is going away
*
* Free up all the regions that belong to this file handle. We use
* the handle as a natural Linux style 'lifetime' indicator and to
* ensure resources are not leaked when their owner explodes in an
* unplanned fashion.
*/
static int memrar_release(struct inode *inode, struct file *filp) static int memrar_release(struct inode *inode, struct file *filp)
{ {
/* Free all regions associated with the given file handle. */ /* Free all regions associated with the given file handle. */
...@@ -733,9 +748,15 @@ static int memrar_release(struct inode *inode, struct file *filp) ...@@ -733,9 +748,15 @@ static int memrar_release(struct inode *inode, struct file *filp)
return 0; return 0;
} }
/* /**
* This function is part of the kernel space memrar driver API. * rar_reserve - reserve RAR memory
* @buffers: buffers to reserve
* @count: number wanted
*
* Reserve a series of buffers in the RAR space. Returns the number of
* buffers successfully allocated
*/ */
size_t rar_reserve(struct RAR_buffer *buffers, size_t count) size_t rar_reserve(struct RAR_buffer *buffers, size_t count)
{ {
struct RAR_buffer * const end = struct RAR_buffer * const end =
...@@ -755,9 +776,14 @@ size_t rar_reserve(struct RAR_buffer *buffers, size_t count) ...@@ -755,9 +776,14 @@ size_t rar_reserve(struct RAR_buffer *buffers, size_t count)
} }
EXPORT_SYMBOL(rar_reserve); EXPORT_SYMBOL(rar_reserve);
/* /**
* This function is part of the kernel space memrar driver API. * rar_release - return RAR buffers
* @buffers: buffers to release
* @size: size of released block
*
* Return a set of buffers to the RAR pool
*/ */
size_t rar_release(struct RAR_buffer *buffers, size_t count) size_t rar_release(struct RAR_buffer *buffers, size_t count)
{ {
struct RAR_buffer * const end = struct RAR_buffer * const end =
...@@ -786,9 +812,16 @@ size_t rar_release(struct RAR_buffer *buffers, size_t count) ...@@ -786,9 +812,16 @@ size_t rar_release(struct RAR_buffer *buffers, size_t count)
} }
EXPORT_SYMBOL(rar_release); EXPORT_SYMBOL(rar_release);
/* /**
* This function is part of the kernel space driver API. * rar_handle_to_bus - RAR to bus address
* @buffers: RAR buffer structure
* @count: number of buffers to convert
*
* Turn a list of RAR handle mappings into actual bus addresses. Note
* that when the device is locked down the bus addresses in question
* are not CPU accessible.
*/ */
size_t rar_handle_to_bus(struct RAR_buffer *buffers, size_t count) size_t rar_handle_to_bus(struct RAR_buffer *buffers, size_t count)
{ {
struct RAR_buffer * const end = struct RAR_buffer * const end =
...@@ -878,43 +911,70 @@ static char const banner[] __initdata = ...@@ -878,43 +911,70 @@ static char const banner[] __initdata =
KERN_INFO KERN_INFO
"Intel RAR Handler: " MEMRAR_VER " initialized.\n"; "Intel RAR Handler: " MEMRAR_VER " initialized.\n";
static int memrar_registration_callback(void *ctx) /**
* memrar_registration_callback - RAR obtained
* @rar: RAR number
*
* We have been granted ownership of the RAR. Add it to our memory
* management tables
*/
static int memrar_registration_callback(unsigned long rar)
{ {
/* /*
* We initialize the RAR parameters early on so that we can * We initialize the RAR parameters early on so that we can
* discontinue memrar device initialization and registration * discontinue memrar device initialization and registration
* if suitably configured RARs are not available. * if suitably configured RARs are not available.
*/ */
int result = memrar_init_rar_resources(memrar_miscdev.name); return memrar_init_rar_resources(rar, memrar_miscdev.name);
}
if (result != 0) /**
return result; * memrar_init - initialise RAR support
*
* Initialise support for RAR handlers. This may get loaded before
* the RAR support is activated, but the callbacks on the registration
* will handle that situation for us anyway.
*/
result = misc_register(&memrar_miscdev); static int __init memrar_init(void)
{
int err;
if (result != 0) { printk(banner);
pr_err("%s: misc_register() failed.\n",
memrar_miscdev.name);
/* Clean up resources previously reserved. */ err = misc_register(&memrar_miscdev);
memrar_fini_rar_resources(); if (err)
} return err;
return result; /* Now claim the two RARs we want */
} err = register_rar(0, memrar_registration_callback, 0);
if (err)
goto fail;
static int __init memrar_init(void) err = register_rar(1, memrar_registration_callback, 1);
{ if (err == 0)
printk(banner); return 0;
return register_rar(&memrar_registration_callback, 0); /* It is possible rar 0 registered and allocated resources then rar 1
failed so do a full resource free */
memrar_fini_rar_resources();
fail:
misc_deregister(&memrar_miscdev);
return err;
} }
/**
* memrar_exit - unregister and unload
*
* Unregister the device and then unload any mappings and release
* the RAR resources
*/
static void __exit memrar_exit(void) static void __exit memrar_exit(void)
{ {
memrar_fini_rar_resources();
misc_deregister(&memrar_miscdev); misc_deregister(&memrar_miscdev);
memrar_fini_rar_resources();
} }
...@@ -925,7 +985,6 @@ module_exit(memrar_exit); ...@@ -925,7 +985,6 @@ module_exit(memrar_exit);
MODULE_AUTHOR("Ossama Othman <ossama.othman@intel.com>"); MODULE_AUTHOR("Ossama Othman <ossama.othman@intel.com>");
MODULE_DESCRIPTION("Intel Restricted Access Region Handler"); MODULE_DESCRIPTION("Intel Restricted Access Region Handler");
MODULE_LICENSE("GPL"); MODULE_LICENSE("GPL");
MODULE_ALIAS_MISCDEV(MISC_DYNAMIC_MINOR);
MODULE_VERSION(MEMRAR_VER); MODULE_VERSION(MEMRAR_VER);
......
...@@ -51,98 +51,159 @@ ...@@ -51,98 +51,159 @@
#include <linux/kernel.h> #include <linux/kernel.h>
/* === Lincroft Message Bus Interface === */ /* === Lincroft Message Bus Interface === */
/* Message Control Register */ #define LNC_MCR_OFFSET 0xD0 /* Message Control Register */
#define LNC_MCR_OFFSET 0xD0 #define LNC_MDR_OFFSET 0xD4 /* Message Data Register */
/* Maximum number of clients (other drivers using this driver) */
#define MAX_RAR_CLIENTS 10
/* Message Data Register */
#define LNC_MDR_OFFSET 0xD4
/* Message Opcodes */ /* Message Opcodes */
#define LNC_MESSAGE_READ_OPCODE 0xD0 #define LNC_MESSAGE_READ_OPCODE 0xD0
#define LNC_MESSAGE_WRITE_OPCODE 0xE0 #define LNC_MESSAGE_WRITE_OPCODE 0xE0
/* Message Write Byte Enables */ /* Message Write Byte Enables */
#define LNC_MESSAGE_BYTE_WRITE_ENABLES 0xF #define LNC_MESSAGE_BYTE_WRITE_ENABLES 0xF
/* B-unit Port */ /* B-unit Port */
#define LNC_BUNIT_PORT 0x3 #define LNC_BUNIT_PORT 0x3
/* === Lincroft B-Unit Registers - Programmed by IA32 firmware === */ /* === Lincroft B-Unit Registers - Programmed by IA32 firmware === */
#define LNC_BRAR0L 0x10 #define LNC_BRAR0L 0x10
#define LNC_BRAR0H 0x11 #define LNC_BRAR0H 0x11
#define LNC_BRAR1L 0x12 #define LNC_BRAR1L 0x12
#define LNC_BRAR1H 0x13 #define LNC_BRAR1H 0x13
/* Reserved for SeP */ /* Reserved for SeP */
#define LNC_BRAR2L 0x14 #define LNC_BRAR2L 0x14
#define LNC_BRAR2H 0x15 #define LNC_BRAR2H 0x15
/* Moorestown supports three restricted access regions. */ /* Moorestown supports three restricted access regions. */
#define MRST_NUM_RAR 3 #define MRST_NUM_RAR 3
/* RAR Bus Address Range */ /* RAR Bus Address Range */
struct RAR_address_range { struct rar_addr {
dma_addr_t low; dma_addr_t low;
dma_addr_t high; dma_addr_t high;
}; };
/* Structure containing low and high RAR register offsets. */ /*
struct RAR_offsets { * We create one of these for each RAR
u32 low; /* Register offset for low RAR bus address. */ */
u32 high; /* Register offset for high RAR bus address. */
};
struct client { struct client {
int (*client_callback)(void *client_data); int (*callback)(unsigned long data);
void *customer_data; unsigned long driver_priv;
int client_called; bool busy;
}; };
static DEFINE_MUTEX(rar_mutex); static DEFINE_MUTEX(rar_mutex);
static DEFINE_MUTEX(lnc_reg_mutex); static DEFINE_MUTEX(lnc_reg_mutex);
struct RAR_device { /*
struct RAR_offsets const rar_offsets[MRST_NUM_RAR]; * One per RAR device (currently only one device)
struct RAR_address_range rar_addr[MRST_NUM_RAR]; */
struct rar_device {
struct rar_addr rar_addr[MRST_NUM_RAR];
struct pci_dev *rar_dev; struct pci_dev *rar_dev;
bool registered; bool registered;
}; bool allocated;
struct client client[MRST_NUM_RAR];
/* this platform has only one rar_device for 3 rar regions */
static struct RAR_device my_rar_device = {
.rar_offsets = {
[0].low = LNC_BRAR0L,
[0].high = LNC_BRAR0H,
[1].low = LNC_BRAR1L,
[1].high = LNC_BRAR1H,
[2].low = LNC_BRAR2L,
[2].high = LNC_BRAR2H
}
}; };
/* this data is for handling requests from other drivers which arrive /* Current platforms have only one rar_device for 3 rar regions */
* prior to this driver initializing static struct rar_device my_rar_device;
/*
* Abstract out multiple device support. Current platforms only
* have a single RAR device.
*/ */
static struct client clients[MAX_RAR_CLIENTS]; /**
static int num_clients; * alloc_rar_device - return a new RAR structure
*
* Return a new (but not yet ready) RAR device object
*/
static struct rar_device *alloc_rar_device(void)
{
if (my_rar_device.allocated)
return NULL;
my_rar_device.allocated = 1;
return &my_rar_device;
}
/* /**
* This function is used to retrieved RAR info using the Lincroft * free_rar_device - free a RAR object
* message bus interface. * @rar: the RAR device being freed
*
* Release a RAR object and any attached resources
*/ */
static int retrieve_rar_addr(struct pci_dev *pdev, static void free_rar_device(struct rar_device *rar)
int offset, {
dma_addr_t *addr) pci_dev_put(rar->rar_dev);
rar->allocated = 0;
}
/**
* _rar_to_device - return the device handling this RAR
* @rar: RAR number
* @off: returned offset
*
* Internal helper for looking up RAR devices. This and alloc are the
* two functions that need touching to go to multiple RAR devices.
*/
static struct rar_device *_rar_to_device(int rar, int *off)
{
if (rar >= 0 && rar <= 3) {
*off = rar;
return &my_rar_device;
}
return NULL;
}
/**
* rar_to_device - return the device handling this RAR
* @rar: RAR number
* @off: returned offset
*
* Return the device this RAR maps to if one is present, otherwise
* returns NULL. Reports the offset relative to the base of this
* RAR device in off.
*/
static struct rar_device *rar_to_device(int rar, int *off)
{
struct rar_device *rar_dev = _rar_to_device(rar, off);
if (rar_dev == NULL || !rar_dev->registered)
return NULL;
return rar_dev;
}
/**
* rar_to_client - return the client handling this RAR
* @rar: RAR number
*
* Return the client this RAR maps to if a mapping is known, otherwise
* returns NULL.
*/
static struct client *rar_to_client(int rar)
{
int idx;
struct rar_device *r = _rar_to_device(rar, &idx);
if (r != NULL)
return &r->client[idx];
return NULL;
}
/**
* rar_read_addr - retrieve a RAR mapping
* @pdev: PCI device for the RAR
* @offset: offset for message
* @addr: returned address
*
* Reads the address of a given RAR register. Returns 0 on success
* or an error code on failure.
*/
static int rar_read_addr(struct pci_dev *pdev, int offset, dma_addr_t *addr)
{ {
/* /*
* ======== The Lincroft Message Bus Interface ======== * ======== The Lincroft Message Bus Interface ========
* Lincroft registers may be obtained from the PCI * Lincroft registers may be obtained via PCI from
* (the Host Bridge) using the Lincroft Message Bus * the host bridge using the Lincroft Message Bus
* Interface. That message bus interface is generally * Interface. That message bus interface is generally
* comprised of two registers: a control register (MCR, 0xDO) * comprised of two registers: a control register (MCR, 0xDO)
* and a data register (MDR, 0xD4). * and a data register (MDR, 0xD4).
...@@ -182,6 +243,7 @@ static int retrieve_rar_addr(struct pci_dev *pdev, ...@@ -182,6 +243,7 @@ static int retrieve_rar_addr(struct pci_dev *pdev,
*/ */
int result; int result;
u32 addr32;
/* Construct control message */ /* Construct control message */
u32 const message = u32 const message =
...@@ -192,11 +254,6 @@ static int retrieve_rar_addr(struct pci_dev *pdev, ...@@ -192,11 +254,6 @@ static int retrieve_rar_addr(struct pci_dev *pdev,
dev_dbg(&pdev->dev, "Offset for 'get' LNC MSG is %x\n", offset); dev_dbg(&pdev->dev, "Offset for 'get' LNC MSG is %x\n", offset);
if (addr == 0) {
WARN_ON(1);
return -EINVAL;
}
/* /*
* We synchronize access to the Lincroft MCR and MDR registers * We synchronize access to the Lincroft MCR and MDR registers
* until BOTH the command is issued through the MCR register * until BOTH the command is issued through the MCR register
...@@ -209,26 +266,25 @@ static int retrieve_rar_addr(struct pci_dev *pdev, ...@@ -209,26 +266,25 @@ static int retrieve_rar_addr(struct pci_dev *pdev,
/* Send the control message */ /* Send the control message */
result = pci_write_config_dword(pdev, LNC_MCR_OFFSET, message); result = pci_write_config_dword(pdev, LNC_MCR_OFFSET, message);
dev_dbg(&pdev->dev, "Result from send ctl register is %x\n", result);
if (!result) { if (!result) {
result = pci_read_config_dword(pdev, LNC_MDR_OFFSET, /* Read back the address as a 32bit value */
(u32 *)addr); result = pci_read_config_dword(pdev, LNC_MDR_OFFSET, &addr32);
dev_dbg(&pdev->dev, *addr = (dma_addr_t)addr32;
"Result from read data register is %x\n", result);
dev_dbg(&pdev->dev,
"Value read from data register is %lx\n",
(unsigned long)*addr);
} }
mutex_unlock(&lnc_reg_mutex); mutex_unlock(&lnc_reg_mutex);
return result; return result;
} }
static int set_rar_address(struct pci_dev *pdev, /**
* rar_set_addr - Set a RAR mapping
* @pdev: PCI device for the RAR
* @offset: offset for message
* @addr: address to set
*
* Sets the address of a given RAR register. Returns 0 on success
* or an error code on failure.
*/
static int rar_set_addr(struct pci_dev *pdev,
int offset, int offset,
dma_addr_t addr) dma_addr_t addr)
{ {
...@@ -236,11 +292,11 @@ static int set_rar_address(struct pci_dev *pdev, ...@@ -236,11 +292,11 @@ static int set_rar_address(struct pci_dev *pdev,
* Data being written to this register must be written before * Data being written to this register must be written before
* writing the appropriate control message to the MCR * writing the appropriate control message to the MCR
* register. * register.
* @note See rar_get_address() for a description of the * See rar_get_addrs() for a description of the
* message bus interface being used here. * message bus interface being used here.
*/ */
int result = 0; int result;
/* Construct control message */ /* Construct control message */
u32 const message = (LNC_MESSAGE_WRITE_OPCODE << 24) u32 const message = (LNC_MESSAGE_WRITE_OPCODE << 24)
...@@ -248,13 +304,6 @@ static int set_rar_address(struct pci_dev *pdev, ...@@ -248,13 +304,6 @@ static int set_rar_address(struct pci_dev *pdev,
| (offset << 8) | (offset << 8)
| (LNC_MESSAGE_BYTE_WRITE_ENABLES << 4); | (LNC_MESSAGE_BYTE_WRITE_ENABLES << 4);
if (addr == 0) {
WARN_ON(1);
return -EINVAL;
}
dev_dbg(&pdev->dev, "Offset for 'set' LNC MSG is %x\n", offset);
/* /*
* We synchronize access to the Lincroft MCR and MDR registers * We synchronize access to the Lincroft MCR and MDR registers
* until BOTH the command is issued through the MCR register * until BOTH the command is issued through the MCR register
...@@ -267,32 +316,27 @@ static int set_rar_address(struct pci_dev *pdev, ...@@ -267,32 +316,27 @@ static int set_rar_address(struct pci_dev *pdev,
/* Send the control message */ /* Send the control message */
result = pci_write_config_dword(pdev, LNC_MDR_OFFSET, addr); result = pci_write_config_dword(pdev, LNC_MDR_OFFSET, addr);
if (!result)
dev_dbg(&pdev->dev, "Result from write data register is %x\n", result); /* And address */
if (!result) {
dev_dbg(&pdev->dev,
"Value written to data register is %lx\n",
(unsigned long)addr);
result = pci_write_config_dword(pdev, LNC_MCR_OFFSET, message); result = pci_write_config_dword(pdev, LNC_MCR_OFFSET, message);
dev_dbg(&pdev->dev, "Result from send ctl register is %x\n",
result);
}
mutex_unlock(&lnc_reg_mutex); mutex_unlock(&lnc_reg_mutex);
return result; return result;
} }
/* /*
* Initialize RAR parameters, such as bus addresses, etc. * rar_init_params - Initialize RAR parameters
*/ * @rar: RAR device to initialise
static int init_rar_params(struct pci_dev *pdev) *
* Initialize RAR parameters, such as bus addresses, etc. Returns 0
* on success, or an error code on failure.
*/
static int init_rar_params(struct rar_device *rar)
{ {
struct pci_dev *pdev = rar->rar_dev;
unsigned int i; unsigned int i;
int result = 0; int result = 0;
int offset = 0x10; /* RAR 0 to 2 in order low/high/low/high/... */
/* Retrieve RAR start and end bus addresses. /* Retrieve RAR start and end bus addresses.
* Access the RAR registers through the Lincroft Message Bus * Access the RAR registers through the Lincroft Message Bus
...@@ -300,15 +344,16 @@ static int init_rar_params(struct pci_dev *pdev) ...@@ -300,15 +344,16 @@ static int init_rar_params(struct pci_dev *pdev)
*/ */
for (i = 0; i < MRST_NUM_RAR; ++i) { for (i = 0; i < MRST_NUM_RAR; ++i) {
struct RAR_offsets const *offset = struct rar_addr *addr = &rar->rar_addr[i];
&my_rar_device.rar_offsets[i];
struct RAR_address_range *addr = &my_rar_device.rar_addr[i]; result = rar_read_addr(pdev, offset++, &addr->low);
if (result != 0)
if ((retrieve_rar_addr(pdev, offset->low, &addr->low) != 0) return result;
|| (retrieve_rar_addr(pdev, offset->high, &addr->high) != 0)) {
result = -1; result = rar_read_addr(pdev, offset++, &addr->high);
break; if (result != 0)
} return result;
/* /*
* Only the upper 22 bits of the RAR addresses are * Only the upper 22 bits of the RAR addresses are
...@@ -336,201 +381,237 @@ static int init_rar_params(struct pci_dev *pdev) ...@@ -336,201 +381,237 @@ static int init_rar_params(struct pci_dev *pdev)
/* Done accessing the device. */ /* Done accessing the device. */
if (result == 0) { if (result == 0) {
int z; for (i = 0; i != MRST_NUM_RAR; ++i) {
for (z = 0; z != MRST_NUM_RAR; ++z) {
/* /*
* "BRAR" refers to the RAR registers in the * "BRAR" refers to the RAR registers in the
* Lincroft B-unit. * Lincroft B-unit.
*/ */
dev_info(&pdev->dev, "BRAR[%u] bus address range = " dev_info(&pdev->dev, "BRAR[%u] bus address range = "
"[%lx, %lx]\n", z, "[%lx, %lx]\n", i,
(unsigned long)my_rar_device.rar_addr[z].low, (unsigned long)rar->rar_addr[i].low,
(unsigned long)my_rar_device.rar_addr[z].high); (unsigned long)rar->rar_addr[i].high);
} }
} }
return result; return result;
} }
/* /**
* The rar_get_address function is used by other device drivers * rar_get_address - get the bus address in a RAR
* to obtain RAR address information on a RAR. It takes three * @start: return value of start address of block
* parameters: * @end: return value of end address of block
* *
* int rar_index * The rar_get_address function is used by other device drivers
* The rar_index is an index to the rar for which you wish to retrieve * to obtain RAR address information on a RAR. It takes three
* the address information. * parameters:
* Values can be 0,1, or 2.
* *
* The function returns a 0 upon success or a -1 if there is no RAR * The function returns a 0 upon success or an error if there is no RAR
* facility on this system. * facility on this system.
*/ */
int rar_get_address(int rar_index, int rar_get_address(int rar_index, dma_addr_t *start, dma_addr_t *end)
dma_addr_t *start_address,
dma_addr_t *end_address)
{ {
int result = -ENODEV; int idx;
struct rar_device *rar = rar_to_device(rar_index, &idx);
if (my_rar_device.registered) {
if (start_address == 0 || end_address == 0 if (rar == NULL) {
|| rar_index >= MRST_NUM_RAR || rar_index < 0) { WARN_ON(1);
result = -EINVAL; return -ENODEV;
} else {
*start_address =
my_rar_device.rar_addr[rar_index].low;
*end_address =
my_rar_device.rar_addr[rar_index].high;
result = 0;
}
} }
return result; *start = rar->rar_addr[idx].low;
*end = rar->rar_addr[idx].high;
return 0;
} }
EXPORT_SYMBOL(rar_get_address); EXPORT_SYMBOL(rar_get_address);
/* /**
* The rar_lock function is ued by other device drivers to lock an RAR. * rar_lock - lock a RAR register
* once an RAR is locked, it stays locked until the next system reboot. * @rar_index: RAR to lock (0-2)
* The function takes one parameter:
* *
* int rar_index * The rar_lock function is ued by other device drivers to lock an RAR.
* The rar_index is an index to the rar that you want to lock. * once a RAR is locked, it stays locked until the next system reboot.
* Values can be 0,1, or 2.
* *
* The function returns a 0 upon success or a -1 if there is no RAR * The function returns a 0 upon success or an error if there is no RAR
* facility on this system. * facility on this system, or the locking fails
*/ */
int rar_lock(int rar_index) int rar_lock(int rar_index)
{ {
int result = -ENODEV; struct rar_device *rar;
int result;
if (rar_index >= MRST_NUM_RAR || rar_index < 0) { int idx;
result = -EINVAL; dma_addr_t low, high;
return result;
}
dev_dbg(&my_rar_device.rar_dev->dev, "rar_lock mutex locking\n");
mutex_lock(&rar_mutex);
if (my_rar_device.registered) { rar = rar_to_device(rar_index, &idx);
dma_addr_t low = my_rar_device.rar_addr[rar_index].low & if (rar == NULL) {
0xfffffc00u; WARN_ON(1);
return -EINVAL;
}
dma_addr_t high = my_rar_device.rar_addr[rar_index].high & low = rar->rar_addr[idx].low & 0xfffffc00u;
0xfffffc00u; high = rar->rar_addr[idx].high & 0xfffffc00u;
/* /*
* Only allow I/O from the graphics and Langwell; * Only allow I/O from the graphics and Langwell;
* Not from the x96 processor * not from the x86 processor
*/ */
if (rar_index == (int)RAR_TYPE_VIDEO) {
low |= 0x00000009;
high |= 0x00000015;
}
else if (rar_index == (int)RAR_TYPE_AUDIO) { if (rar_index == RAR_TYPE_VIDEO) {
/* Only allow I/O from Langwell; nothing from x86 */ low |= 0x00000009;
low |= 0x00000008; high |= 0x00000015;
high |= 0x00000018; } else if (rar_index == RAR_TYPE_AUDIO) {
} /* Only allow I/O from Langwell; nothing from x86 */
low |= 0x00000008;
high |= 0x00000018;
} else
/* Read-only from all agents */
high |= 0x00000018;
else /*
/* Read-only from all agents */ * Now program the register using the Lincroft message
high |= 0x00000018; * bus interface.
*/
result = rar_set_addr(rar->rar_dev,
2 * idx, low);
/* if (result == 0)
* Now program the register using the Lincroft message result = rar_set_addr(rar->rar_dev,
* bus interface. 2 * idx + 1, high);
*/
result = set_rar_address(my_rar_device.rar_dev,
my_rar_device.rar_offsets[rar_index].low,
low);
if (result == 0)
result = set_rar_address(
my_rar_device.rar_dev,
my_rar_device.rar_offsets[rar_index].high,
high);
}
dev_dbg(&my_rar_device.rar_dev->dev, "rar_lock mutex unlocking\n");
mutex_unlock(&rar_mutex);
return result; return result;
} }
EXPORT_SYMBOL(rar_lock); EXPORT_SYMBOL(rar_lock);
/* The register_rar function is to used by other device drivers /**
* to ensure that this driver is ready. As we cannot be sure of * register_rar - register a RAR handler
* the compile/execute order of dirvers in ther kernel, it is * @num: RAR we wish to register for
* best to give this driver a callback function to call when * @callback: function to call when RAR support is available
* it is ready to give out addresses. The callback function * @data: data to pass to this function
* would have those steps that continue the initialization of *
* a driver that do require a valid RAR address. One of those * The register_rar function is to used by other device drivers
* steps would be to call rar_get_address() * to ensure that this driver is ready. As we cannot be sure of
* This function return 0 on success an -1 on failure. * the compile/execute order of drivers in ther kernel, it is
*/ * best to give this driver a callback function to call when
int register_rar(int (*callback)(void *yourparameter), void *yourparameter) * it is ready to give out addresses. The callback function
* would have those steps that continue the initialization of
* a driver that do require a valid RAR address. One of those
* steps would be to call rar_get_address()
*
* This function return 0 on success an error code on failure.
*/
int register_rar(int num, int (*callback)(unsigned long data),
unsigned long data)
{ {
/* For now we hardcode a single RAR device */
int result = -ENODEV; struct rar_device *rar;
struct client *c;
if (callback == NULL) int idx;
return -EINVAL; int retval = 0;
mutex_lock(&rar_mutex); mutex_lock(&rar_mutex);
if (my_rar_device.registered) { /* Do we have a client mapping for this RAR number ? */
c = rar_to_client(num);
if (c == NULL) {
retval = -ERANGE;
goto done;
}
/* Is it claimed ? */
if (c->busy) {
retval = -EBUSY;
goto done;
}
c->busy = 1;
/* See if we have a handler for this RAR yet, if we do then fire it */
rar = rar_to_device(num, &idx);
mutex_unlock(&rar_mutex); if (rar) {
/* /*
* if the driver already registered, then we can simply * if the driver already registered, then we can simply
* call the callback right now * call the callback right now
*/ */
(*callback)(data);
return (*callback)(yourparameter); goto done;
}
if (num_clients < MRST_NUM_RAR) {
clients[num_clients].client_callback = callback;
clients[num_clients].customer_data = yourparameter;
num_clients += 1;
result = 0;
} }
/* Arrange to be called back when the hardware is found */
c->callback = callback;
c->driver_priv = data;
done:
mutex_unlock(&rar_mutex); mutex_unlock(&rar_mutex);
return result; return retval;
} }
EXPORT_SYMBOL(register_rar); EXPORT_SYMBOL(register_rar);
/* Suspend - returns -ENOSYS */ /**
static int rar_suspend(struct pci_dev *dev, pm_message_t state) * unregister_rar - release a RAR allocation
* @num: RAR number
*
* Releases a RAR allocation, or pending allocation. If a callback is
* pending then this function will either complete before the unregister
* returns or not at all.
*/
void unregister_rar(int num)
{ {
return -ENOSYS; struct client *c;
mutex_lock(&rar_mutex);
c = rar_to_client(num);
if (c == NULL || !c->busy)
WARN_ON(1);
else
c->busy = 0;
mutex_unlock(&rar_mutex);
} }
EXPORT_SYMBOL(unregister_rar);
static int rar_resume(struct pci_dev *dev) /**
* rar_callback - Process callbacks
* @rar: new RAR device
*
* Process the callbacks for a newly found RAR device.
*/
static void rar_callback(struct rar_device *rar)
{ {
return -ENOSYS; struct client *c = &rar->client[0];
int i;
mutex_lock(&rar_mutex);
rar->registered = 1; /* Ensure no more callbacks queue */
for (i = 0; i < MRST_NUM_RAR; i++) {
if (c->callback && c->busy) {
c->callback(c->driver_priv);
c->callback = NULL;
}
c++;
}
mutex_unlock(&rar_mutex);
} }
/* /**
* This function registers the driver with the device subsystem ( * rar_probe - PCI probe callback
* either PCI, USB, etc). * @dev: PCI device
* Function that is activaed on the succesful probe of the RAR device * @id: matching entry in the match table
* (Moorestown host controller). *
* A RAR device has been discovered. Initialise it and if successful
* process any pending callbacks that can now be completed.
*/ */
static int rar_probe(struct pci_dev *dev, const struct pci_device_id *id) static int rar_probe(struct pci_dev *dev, const struct pci_device_id *id)
{ {
int error; int error;
int counter; struct rar_device *rar;
dev_dbg(&dev->dev, "PCI probe starting\n"); dev_dbg(&dev->dev, "PCI probe starting\n");
/* enable the device */ rar = alloc_rar_device();
if (rar == NULL)
return -EBUSY;
/* Enable the device */
error = pci_enable_device(dev); error = pci_enable_device(dev);
if (error) { if (error) {
dev_err(&dev->dev, dev_err(&dev->dev,
...@@ -538,50 +619,30 @@ static int rar_probe(struct pci_dev *dev, const struct pci_device_id *id) ...@@ -538,50 +619,30 @@ static int rar_probe(struct pci_dev *dev, const struct pci_device_id *id)
goto end_function; goto end_function;
} }
/* we have only one device; fill in the rar_device structure */ /* Fill in the rar_device structure */
my_rar_device.rar_dev = dev; rar->rar_dev = pci_dev_get(dev);
pci_set_drvdata(dev, rar);
/* /*
* Initialize the RAR parameters, which have to be retrieved * Initialize the RAR parameters, which have to be retrieved
* via the message bus interface. * via the message bus interface.
*/ */
error = init_rar_params(dev); error = init_rar_params(rar);
if (error) { if (error) {
pci_disable_device(dev); pci_disable_device(dev);
dev_err(&dev->dev, "Error retrieving RAR addresses\n");
dev_err(&dev->dev,
"Error retrieving RAR addresses\n");
goto end_function; goto end_function;
} }
dev_dbg(&dev->dev, "PCI probe locking\n");
mutex_lock(&rar_mutex);
my_rar_device.registered = 1;
/* now call anyone who has registered (using callbacks) */ /* now call anyone who has registered (using callbacks) */
for (counter = 0; counter < num_clients; counter += 1) { rar_callback(rar);
if (clients[counter].client_callback) { return 0;
error = (*clients[counter].client_callback)(
clients[counter].customer_data);
/* set callback to NULL to indicate it has been done */
clients[counter].client_callback = NULL;
dev_dbg(&my_rar_device.rar_dev->dev,
"Callback called for %d\n",
counter);
}
}
dev_dbg(&dev->dev, "PCI probe unlocking\n");
mutex_unlock(&rar_mutex);
end_function: end_function:
free_rar_device(rar);
return error; return error;
} }
const struct pci_device_id rar_pci_id_tbl[] = { const struct pci_device_id rar_pci_id_tbl[] = {
{ PCI_DEVICE(PCI_VENDOR_ID_INTEL, PCI_RAR_DEVICE_ID) }, { PCI_VDEVICE(INTEL, 0x4110) },
{ 0 } { 0 }
}; };
...@@ -594,8 +655,7 @@ static struct pci_driver rar_pci_driver = { ...@@ -594,8 +655,7 @@ static struct pci_driver rar_pci_driver = {
.name = "rar_register_driver", .name = "rar_register_driver",
.id_table = rar_pci_id_tbl, .id_table = rar_pci_id_tbl,
.probe = rar_probe, .probe = rar_probe,
.suspend = rar_suspend, /* Cannot be unplugged - no remove */
.resume = rar_resume
}; };
static int __init rar_init_handler(void) static int __init rar_init_handler(void)
......
...@@ -21,63 +21,23 @@ ...@@ -21,63 +21,23 @@
#ifndef _RAR_REGISTER_H #ifndef _RAR_REGISTER_H
#define _RAR_REGISTER_H #define _RAR_REGISTER_H
# include <linux/types.h> #include <linux/types.h>
/* following are used both in drivers as well as user space apps */ /* following are used both in drivers as well as user space apps */
enum RAR_type {
RAR_TYPE_VIDEO = 0,
RAR_TYPE_AUDIO,
RAR_TYPE_IMAGE,
RAR_TYPE_DATA
};
#ifdef __KERNEL__ #define RAR_TYPE_VIDEO 0
#define RAR_TYPE_AUDIO 1
#define RAR_TYPE_IMAGE 2
#define RAR_TYPE_DATA 3
/* PCI device id for controller */ #ifdef __KERNEL__
#define PCI_RAR_DEVICE_ID 0x4110
/* The register_rar function is to used by other device drivers struct rar_device;
* to ensure that this driver is ready. As we cannot be sure of
* the compile/execute order of dirvers in ther kernel, it is
* best to give this driver a callback function to call when
* it is ready to give out addresses. The callback function
* would have those steps that continue the initialization of
* a driver that do require a valid RAR address. One of those
* steps would be to call get_rar_address()
* This function return 0 on success an -1 on failure.
*/
int register_rar(int (*callback)(void *yourparameter), void *yourparameter);
/* The get_rar_address function is used by other device drivers int register_rar(int num,
* to obtain RAR address information on a RAR. It takes two int (*callback)(unsigned long data), unsigned long data);
* parameter: void unregister_rar(int num);
* int rar_get_address(int rar_index, dma_addr_t *start, dma_addr_t *end);
* int rar_index
* The rar_index is an index to the rar for which you wish to retrieve
* the address information.
* Values can be 0,1, or 2.
*
* struct RAR_address_struct is a pointer to a place to which the function
* can return the address structure for the RAR.
*
* The function returns a 0 upon success or a -1 if there is no RAR
* facility on this system.
*/
int rar_get_address(int rar_index,
dma_addr_t *start_address,
dma_addr_t *end_address);
/* The lock_rar function is ued by other device drivers to lock an RAR.
* once an RAR is locked, it stays locked until the next system reboot.
* The function takes one parameter:
*
* int rar_index
* The rar_index is an index to the rar that you want to lock.
* Values can be 0,1, or 2.
*
* The function returns a 0 upon success or a -1 if there is no RAR
* facility on this system.
*/
int rar_lock(int rar_index); int rar_lock(int rar_index);
#endif /* __KERNEL__ */ #endif /* __KERNEL__ */
......
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