Commit 63cc932b authored by Alex Elder's avatar Alex Elder Committed by Greg Kroah-Hartman

greybus: October 1 updates

Update the definitions in "greybus_manifest.h" to reflect the
changes to the Greybus specification made on October 1.

They are:
    - renaming "device" to be "interface"
    - renumbering greybus descriptor type
    - eliminating the notion of a "function"
    - defining a CPort's protocol in the CPort descriptor
    - having a "class" take on the types previously used for "function"
    - renaming "serial number" to be "unique id" (for now)
    - relying on an interface's maximum cport id to determine how
      much device+cport address space the interface consumes
    - adding a simple class descriptor
    - renaming gb_interface->interface_id to be gb_interface->id

This also reorders some things to match ordering in the document,
and adds some commentary for the various structures.

Since greybus_function_type is gone, we eliminate the "type" field
from a function structure.  (Functions are going away, next.)
Signed-off-by: default avatarAlex Elder <elder@linaro.org>
Signed-off-by: default avatarGreg Kroah-Hartman <greg@kroah.com>
parent b05890db
...@@ -22,7 +22,7 @@ static DEFINE_SPINLOCK(gb_functions_lock); ...@@ -22,7 +22,7 @@ static DEFINE_SPINLOCK(gb_functions_lock);
* pointer if a failure occurs due to memory exhaustion. * pointer if a failure occurs due to memory exhaustion.
*/ */
struct gb_function *gb_function_create(struct gb_interface *interface, struct gb_function *gb_function_create(struct gb_interface *interface,
u16 cport_id, enum greybus_function_type type) u16 cport_id)
{ {
struct gb_function *function; struct gb_function *function;
...@@ -32,7 +32,6 @@ struct gb_function *gb_function_create(struct gb_interface *interface, ...@@ -32,7 +32,6 @@ struct gb_function *gb_function_create(struct gb_interface *interface,
function->interface = interface; /* XXX refcount? */ function->interface = interface; /* XXX refcount? */
function->cport_id = cport_id; function->cport_id = cport_id;
function->type = type;
spin_lock_irq(&gb_functions_lock); spin_lock_irq(&gb_functions_lock);
list_add_tail(&function->links, &interface->functions); list_add_tail(&function->links, &interface->functions);
......
...@@ -12,14 +12,12 @@ ...@@ -12,14 +12,12 @@
struct gb_function { struct gb_function {
struct gb_interface *interface; struct gb_interface *interface;
u16 cport_id; u16 cport_id;
enum greybus_function_type type;
struct list_head links; /* interface->functions */ struct list_head links; /* interface->functions */
}; };
struct gb_function *gb_function_create(struct gb_interface *interface, struct gb_function *gb_function_create(struct gb_interface *interface,
u16 cport_id, u16 cport_id);
enum greybus_function_type function_type);
void gb_function_destroy(struct gb_function *function); void gb_function_destroy(struct gb_function *function);
#endif /* __FUNCTION_H */ #endif /* __FUNCTION_H */
...@@ -13,7 +13,7 @@ struct greybus_module_id { ...@@ -13,7 +13,7 @@ struct greybus_module_id {
__u16 match_flags; __u16 match_flags;
__u16 vendor; __u16 vendor;
__u16 product; __u16 product;
__u64 serial_number; __u64 unique_id;
kernel_ulong_t driver_info kernel_ulong_t driver_info
__attribute__((aligned(sizeof(kernel_ulong_t)))); __attribute__((aligned(sizeof(kernel_ulong_t))));
......
...@@ -17,13 +17,24 @@ ...@@ -17,13 +17,24 @@
enum greybus_descriptor_type { enum greybus_descriptor_type {
GREYBUS_TYPE_INVALID = 0x00, GREYBUS_TYPE_INVALID = 0x00,
GREYBUS_TYPE_MODULE = 0x01, GREYBUS_TYPE_MODULE = 0x01,
GREYBUS_TYPE_DEVICE = 0x02, GREYBUS_TYPE_STRING = 0x02,
GREYBUS_TYPE_CLASS = 0x03, GREYBUS_TYPE_INTERFACE = 0x03,
GREYBUS_TYPE_STRING = 0x04, GREYBUS_TYPE_CPORT = 0x04,
GREYBUS_TYPE_CPORT = 0x05, GREYBUS_TYPE_CLASS = 0x05,
}; };
enum greybus_function_type { enum greybus_protocol {
GREYBUS_PROTOCOL_CONTROL = 0x00,
GREYBUS_PROTOCOL_AP = 0x01,
GREYBUS_PROTOCOL_GPIO = 0x02,
GREYBUS_PROTOCOL_I2C = 0x03,
GREYBUS_PROTOCOL_UART = 0x04,
GREYBUS_PROTOCOL_HID = 0x05,
/* ... */
GREYBUS_PROTOCOL_VENDOR = 0xff,
};
enum greybus_class_type {
GREYBUS_FUNCTION_CONTROL = 0x00, GREYBUS_FUNCTION_CONTROL = 0x00,
GREYBUS_FUNCTION_USB = 0x01, GREYBUS_FUNCTION_USB = 0x01,
GREYBUS_FUNCTION_GPIO = 0x02, GREYBUS_FUNCTION_GPIO = 0x02,
...@@ -48,57 +59,71 @@ struct greybus_descriptor_module { ...@@ -48,57 +59,71 @@ struct greybus_descriptor_module {
__le16 vendor; __le16 vendor;
__le16 product; __le16 product;
__le16 version; __le16 version;
__le64 serial_number;
__u8 vendor_stringid; __u8 vendor_stringid;
__u8 product_stringid; __u8 product_stringid;
__le64 unique_id;
};
/*
* The string in a string descriptor is not NUL-terminated. The
* size of the descriptor will be rounded up to a multiple of 4
* bytes, by padding the string with 0x00 bytes if necessary.
*/
struct greybus_descriptor_string {
__u8 length;
__u8 id;
__u8 string[0];
}; };
/* /*
* A UniPro device normally supports a range of 32 CPorts (0..31). * An interface descriptor simply defines a module-unique id for
* It is possible to support more than this by having a UniPro * each interface present on a module. Its sole purpose is to allow
* switch treat one device as if it were more than one. E.g., * CPort descriptors to specify which interface they are associated
* allocate 3 device ids (rather than the normal--1) to physical * with. Normally there's only one interface, with id 0. The
* device 5, and configure the switch to route all packets destined * second one must have id 1, and so on consecutively.
* for "encoded" device ids 5, 6, and 7 to physical device 5.
* Device 5 uses the encoded device id in incoming UniPro packets to
* determine which bank of 32 CPorts should receive the UniPro
* segment.
* *
* The "scale" field in this structure is used to define the number * The largest CPort id associated with an interface (defined by a
* of encoded device ids should be allocated for this physical * CPort descriptor in the manifest) is used to determine how to
* device. Scale is normally 1, to represent 32 available CPorts. * encode the device id and module number in UniPro packets
* A scale value 2 represents up to 64 CPorts; scale value 3 * that use the interface.
* represents up to 96 CPorts, and so on.
*/ */
struct greybus_descriptor_interface { struct greybus_descriptor_interface {
__u8 id; /* module-relative id (0..) */ __u8 id; /* module-relative id (0..) */
__u8 scale; /* indicates range of of CPorts supported */
/* UniPro gear, number of in/out lanes */
}; };
/*
* A CPort descriptor indicates the id of the interface within the
* module it's associated with, along with the CPort id used to
* address the CPort. The protocol defines the format of messages
* exchanged using the CPort.
*/
struct greybus_descriptor_cport { struct greybus_descriptor_cport {
__u8 interface;
__le16 id; __le16 id;
__u8 function_type; /* enum greybus_function_type */ __u8 protocol; /* enum greybus_protocol */
}; };
struct greybus_descriptor_string { /*
__u8 length; * A class descriptor defines functionality supplied by a module.
__u8 id; * Beyond that, not much else is defined yet...
__u8 string[0]; */
struct greybus_descriptor_class {
__u8 class; /* enum greybus_class_type */
}; };
struct greybus_descriptor_header { struct greybus_descriptor_header {
__le16 size; __le16 size;
__u8 type; /* enum greybus_descriptor_type */ __u8 type; /* enum greybus_descriptor_type */
}; };
struct greybus_descriptor { struct greybus_descriptor {
struct greybus_descriptor_header header; struct greybus_descriptor_header header;
union { union {
struct greybus_descriptor_module module; struct greybus_descriptor_module module;
struct greybus_descriptor_string string; struct greybus_descriptor_string string;
struct greybus_descriptor_interface interface; struct greybus_descriptor_interface interface;
struct greybus_descriptor_cport cport; struct greybus_descriptor_cport cport;
struct greybus_descriptor_class class;
}; };
}; };
......
...@@ -32,7 +32,7 @@ gb_interface_create(struct gb_module *gmod, u8 interface_id) ...@@ -32,7 +32,7 @@ gb_interface_create(struct gb_module *gmod, u8 interface_id)
return NULL; return NULL;
interface->gmod = gmod; /* XXX refcount? */ interface->gmod = gmod; /* XXX refcount? */
interface->interface_id = interface_id; interface->id = interface_id;
INIT_LIST_HEAD(&interface->functions); INIT_LIST_HEAD(&interface->functions);
spin_lock_irq(&gb_interfaces_lock); spin_lock_irq(&gb_interfaces_lock);
......
...@@ -13,7 +13,7 @@ ...@@ -13,7 +13,7 @@
struct gb_interface { struct gb_interface {
struct gb_module *gmod; struct gb_module *gmod;
u8 interface_id; u8 id;
struct list_head functions; struct list_head functions;
struct list_head links; /* module->interfaces */ struct list_head links; /* module->interfaces */
......
...@@ -80,11 +80,6 @@ static int identify_descriptor(struct greybus_descriptor *desc, size_t size) ...@@ -80,11 +80,6 @@ static int identify_descriptor(struct greybus_descriptor *desc, size_t size)
return -EINVAL; return -EINVAL;
} }
break; break;
case GREYBUS_TYPE_DEVICE:
break;
case GREYBUS_TYPE_CLASS:
pr_err("class descriptor found (ignoring)\n");
break;
case GREYBUS_TYPE_STRING: case GREYBUS_TYPE_STRING:
expected_size = sizeof(struct greybus_descriptor_header); expected_size = sizeof(struct greybus_descriptor_header);
expected_size += sizeof(struct greybus_descriptor_string); expected_size += sizeof(struct greybus_descriptor_string);
...@@ -95,6 +90,8 @@ static int identify_descriptor(struct greybus_descriptor *desc, size_t size) ...@@ -95,6 +90,8 @@ static int identify_descriptor(struct greybus_descriptor *desc, size_t size)
return -EINVAL; return -EINVAL;
} }
break; break;
case GREYBUS_TYPE_INTERFACE:
break;
case GREYBUS_TYPE_CPORT: case GREYBUS_TYPE_CPORT:
if (desc_size < sizeof(struct greybus_descriptor_cport)) { if (desc_size < sizeof(struct greybus_descriptor_cport)) {
pr_err("cport descriptor too small (%u)\n", pr_err("cport descriptor too small (%u)\n",
...@@ -102,6 +99,9 @@ static int identify_descriptor(struct greybus_descriptor *desc, size_t size) ...@@ -102,6 +99,9 @@ static int identify_descriptor(struct greybus_descriptor *desc, size_t size)
return -EINVAL; return -EINVAL;
} }
break; break;
case GREYBUS_TYPE_CLASS:
pr_warn("class descriptor found (ignoring)\n");
break;
case GREYBUS_TYPE_INVALID: case GREYBUS_TYPE_INVALID:
default: default:
pr_err("invalid descriptor type (%hhu)\n", desc_header->type); pr_err("invalid descriptor type (%hhu)\n", desc_header->type);
...@@ -183,7 +183,7 @@ u32 gb_manifest_parse_cports(struct gb_interface *interface) ...@@ -183,7 +183,7 @@ u32 gb_manifest_parse_cports(struct gb_interface *interface)
while (true) { while (true) {
struct manifest_desc *descriptor; struct manifest_desc *descriptor;
struct greybus_descriptor_cport *desc_cport; struct greybus_descriptor_cport *desc_cport;
enum greybus_function_type function_type; enum greybus_protocol protocol;
u16 cport_id; u16 cport_id;
bool found; bool found;
...@@ -191,19 +191,20 @@ u32 gb_manifest_parse_cports(struct gb_interface *interface) ...@@ -191,19 +191,20 @@ u32 gb_manifest_parse_cports(struct gb_interface *interface)
found = false; found = false;
list_for_each_entry(descriptor, &manifest_descs, links) { list_for_each_entry(descriptor, &manifest_descs, links) {
if (descriptor->type == GREYBUS_TYPE_CPORT) { if (descriptor->type == GREYBUS_TYPE_CPORT) {
found = true; desc_cport = descriptor->data;
break; if (desc_cport->interface == interface->id) {
found = true;
break;
}
} }
} }
if (!found) if (!found)
break; break;
/* Found one. Set up its function structure */ /* Found one. Set up its function structure */
desc_cport = descriptor->data; protocol = (enum greybus_protocol)desc_cport->protocol;
function_type =
(enum greybus_function_type)desc_cport->function_type;
cport_id = le16_to_cpu(desc_cport->id); cport_id = le16_to_cpu(desc_cport->id);
if (!gb_function_create(interface, cport_id, function_type)) if (!gb_function_create(interface, cport_id))
return 0; /* Error */ return 0; /* Error */
count++; count++;
...@@ -231,7 +232,7 @@ static u32 gb_manifest_parse_interfaces(struct gb_module *gmod) ...@@ -231,7 +232,7 @@ static u32 gb_manifest_parse_interfaces(struct gb_module *gmod)
/* Find an interface descriptor */ /* Find an interface descriptor */
list_for_each_entry(descriptor, &manifest_descs, links) { list_for_each_entry(descriptor, &manifest_descs, links) {
if (descriptor->type == GREYBUS_TYPE_DEVICE) { if (descriptor->type == GREYBUS_TYPE_INTERFACE) {
found = true; found = true;
break; break;
} }
...@@ -284,7 +285,7 @@ struct gb_module *gb_manifest_parse_module(struct manifest_desc *module_desc) ...@@ -284,7 +285,7 @@ struct gb_module *gb_manifest_parse_module(struct manifest_desc *module_desc)
gmod->vendor = le16_to_cpu(desc_module->vendor); gmod->vendor = le16_to_cpu(desc_module->vendor);
gmod->product = le16_to_cpu(desc_module->product); gmod->product = le16_to_cpu(desc_module->product);
gmod->version = le16_to_cpu(desc_module->version); gmod->version = le16_to_cpu(desc_module->version);
gmod->serial_number = le64_to_cpu(desc_module->serial_number); gmod->unique_id = le64_to_cpu(desc_module->unique_id);
/* Release the module descriptor, now that we're done with it */ /* Release the module descriptor, now that we're done with it */
release_manifest_descriptor(module_desc); release_manifest_descriptor(module_desc);
......
...@@ -23,7 +23,7 @@ static int gb_module_match_one_id(struct gb_module *gmod, ...@@ -23,7 +23,7 @@ static int gb_module_match_one_id(struct gb_module *gmod,
return 0; return 0;
if ((id->match_flags & GREYBUS_DEVICE_ID_MATCH_SERIAL) && if ((id->match_flags & GREYBUS_DEVICE_ID_MATCH_SERIAL) &&
(id->serial_number != gmod->serial_number)) (id->unique_id != gmod->unique_id))
return 0; return 0;
return 1; return 1;
...@@ -35,7 +35,7 @@ const struct greybus_module_id *gb_module_match_id(struct gb_module *gmod, ...@@ -35,7 +35,7 @@ const struct greybus_module_id *gb_module_match_id(struct gb_module *gmod,
if (id == NULL) if (id == NULL)
return NULL; return NULL;
for (; id->vendor || id->product || id->serial_number || for (; id->vendor || id->product || id->unique_id ||
id->driver_info; id++) { id->driver_info; id++) {
if (gb_module_match_one_id(gmod, id)) if (gb_module_match_one_id(gmod, id))
return id; return id;
......
...@@ -24,9 +24,9 @@ struct gb_module { ...@@ -24,9 +24,9 @@ struct gb_module {
u16 vendor; u16 vendor;
u16 product; u16 product;
u16 version; u16 version;
u64 serial_number;
char *vendor_string; char *vendor_string;
char *product_string; char *product_string;
u64 unique_id;
int num_cports; int num_cports;
int num_strings; int num_strings;
......
...@@ -40,7 +40,7 @@ static ssize_t module_serial_number_show(struct device *dev, ...@@ -40,7 +40,7 @@ static ssize_t module_serial_number_show(struct device *dev,
{ {
struct gb_module *gmod = to_gb_module(dev); struct gb_module *gmod = to_gb_module(dev);
return sprintf(buf, "%llX\n", (unsigned long long)gmod->serial_number); return sprintf(buf, "%llX\n", (unsigned long long)gmod->unique_id);
} }
static DEVICE_ATTR_RO(module_serial_number); static DEVICE_ATTR_RO(module_serial_number);
...@@ -86,7 +86,7 @@ static umode_t module_attrs_are_visible(struct kobject *kobj, ...@@ -86,7 +86,7 @@ static umode_t module_attrs_are_visible(struct kobject *kobj,
return mode; return mode;
if (gmod->vendor || gmod->product || gmod->version) if (gmod->vendor || gmod->product || gmod->version)
return mode; return mode;
if (gmod->serial_number) if (gmod->unique_id)
return mode; return mode;
return 0; return 0;
......
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