Commit a239f67c authored by Greg Kroah-Hartman's avatar Greg Kroah-Hartman

greybus: start parsing descriptor structures

parent d94a44a5
...@@ -12,6 +12,7 @@ ...@@ -12,6 +12,7 @@
#include <linux/module.h> #include <linux/module.h>
#include <linux/moduleparam.h> #include <linux/moduleparam.h>
#include <linux/kernel.h> #include <linux/kernel.h>
#include <linux/slab.h>
#include <linux/kthread.h> #include <linux/kthread.h>
#include <linux/device.h> #include <linux/device.h>
...@@ -155,8 +156,8 @@ void greybus_deregister(struct greybus_driver *driver) ...@@ -155,8 +156,8 @@ void greybus_deregister(struct greybus_driver *driver)
EXPORT_SYMBOL_GPL(greybus_deregister); EXPORT_SYMBOL_GPL(greybus_deregister);
int new_device(struct greybus_device *gdev, static int gb_init_subdevs(struct greybus_device *gdev,
const struct greybus_module_id *id) const struct greybus_module_id *id)
{ {
int retval; int retval;
...@@ -191,6 +192,118 @@ int new_device(struct greybus_device *gdev, ...@@ -191,6 +192,118 @@ int new_device(struct greybus_device *gdev,
return retval; return retval;
} }
static const struct greybus_module_id fake_gb_id =
{ GREYBUS_DEVICE(0x42, 0x42) };
/**
* greybus_new_device:
*
* Pass in a buffer that _should_ be a set of greybus descriptor fields and spit
* out a greybus device structure.
*/
struct greybus_device *greybus_new_device(int module_number, u8 *data, int size)
{
struct greybus_device *gdev;
struct greybus_descriptor_block_header *block;
struct greybus_descriptor *desc;
int retval;
int overall_size;
int header_size;
int desc_size;
u8 version_major;
u8 version_minor;
/* we have to have at _least_ the block header */
if (size <= sizeof(struct greybus_descriptor_block_header))
return NULL;
gdev = kzalloc(sizeof(*gdev), GFP_KERNEL);
if (!gdev)
return NULL;
gdev->module_number = module_number;
block = (struct greybus_descriptor_block_header *)data;
overall_size = le16_to_cpu(block->size);
if (overall_size != size) {
pr_err("size != block header size, %d != %d\n", size,
overall_size);
goto error;
}
version_major = block->version_major;
version_minor = block->version_minor;
// FIXME - check version major/minor here!
size -= sizeof(struct greybus_descriptor_block_header);
data += sizeof(struct greybus_descriptor_block_header);
while (size > 0) {
desc = (struct greybus_descriptor *)data;
desc_size = le16_to_cpu(desc->header.size);
switch (desc->header.type) {
case GREYBUS_TYPE_FUNCTION:
header_size =
sizeof(struct greybus_descriptor_function);
if (desc_size != header_size) {
pr_err("invalid function header size %d\n",
desc_size);
goto error;
}
memcpy(&gdev->function, &desc->function, header_size);
size -= header_size;
data += header_size;
break;
case GREYBUS_TYPE_MODULE_ID:
header_size =
sizeof(struct greybus_descriptor_module_id);
if (desc_size != header_size) {
pr_err("invalid module header size %d\n",
desc_size);
goto error;
}
memcpy(&gdev->module_id, &desc->module_id, header_size);
size -= header_size;
data += header_size;
break;
case GREYBUS_TYPE_SERIAL_NUMBER:
header_size =
sizeof(struct greybus_descriptor_serial_number);
if (desc_size != header_size) {
pr_err("invalid serial number header size %d\n",
desc_size);
goto error;
}
memcpy(&gdev->serial_number, &desc->serial_number,
header_size);
size -= header_size;
data += header_size;
break;
case GREYBUS_TYPE_DEVICE_STRING:
case GREYBUS_TYPE_CPORT:
case GREYBUS_TYPE_INVALID:
default:
pr_err("invalid descriptor type %d\n", desc->header.type);
goto error;
}
#if 0
struct greybus_descriptor_string string;
struct greybus_descriptor_cport cport;
#endif
}
retval = gb_init_subdevs(gdev, &fake_gb_id);
if (retval)
goto error;
return gdev;
error:
kfree(gdev);
return NULL;
}
void remove_device(struct greybus_device *gdev) void remove_device(struct greybus_device *gdev)
{ {
/* tear down all of the "sub device types" for this device */ /* tear down all of the "sub device types" for this device */
......
...@@ -37,6 +37,7 @@ struct gbuf; ...@@ -37,6 +37,7 @@ struct gbuf;
struct cport { struct cport {
u16 number; u16 number;
u16 size;
// FIXME, what else? // FIXME, what else?
}; };
...@@ -87,11 +88,12 @@ struct gb_usb_device; ...@@ -87,11 +88,12 @@ struct gb_usb_device;
struct greybus_device { struct greybus_device {
struct device dev; struct device dev;
u16 module_number;
struct greybus_descriptor_function function; struct greybus_descriptor_function function;
struct greybus_descriptor_module_id module_id; struct greybus_descriptor_module_id module_id;
struct greybus_descriptor_serial_number serial_number; struct greybus_descriptor_serial_number serial_number;
int num_cport; int num_cport;
struct cport cport[0]; struct cport *cport[10]; // FIXME - no more than 10 cports per device...
struct gb_i2c_device *gb_i2c_dev; struct gb_i2c_device *gb_i2c_dev;
struct gb_gpio_device *gb_gpio_dev; struct gb_gpio_device *gb_gpio_dev;
......
...@@ -10,7 +10,7 @@ ...@@ -10,7 +10,7 @@
#ifndef __GREYBUS_DESC_H #ifndef __GREYBUS_DESC_H
#define __GREYBUS_DESC_H #define __GREYBUS_DESC_H
struct greybus_decriptor_block_header { struct greybus_descriptor_block_header {
__le16 size; __le16 size;
__u8 version_major; __u8 version_major;
__u8 version_minor; __u8 version_minor;
......
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