Commit 22b320f4 authored by Alex Elder's avatar Alex Elder Committed by Greg Kroah-Hartman

greybus: add response buffer to an operation

We need to track both request messages and response messages in
operations.  So add another gbuf (and payload pointer) field to
the operation structure, and rename them to indicate which one
is which.  Allow the creator specify the size of the response
buffer; just leave it a null pointer if the size is 0.

Define a new helper function gb_operation_gbuf_create() to
encapsulate creating either a request or a response buffer.

Any buffer associated with a connection will (eventually) have been
created as part of an operation.  So stash the operation pointer in
the gbuf as the context pointer.  Whether a buffer is for the
request or the response can be determined by pointer comparison.
Signed-off-by: default avatarAlex Elder <elder@linaro.org>
Signed-off-by: default avatarGreg Kroah-Hartman <greg@kroah.com>
parent 17d265f6
......@@ -13,6 +13,12 @@
#include "greybus.h"
/*
* The top bit of the type in an operation message header indicates
* whether the message is a request (bit clear) or response (bit set)
*/
#define GB_OPERATION_TYPE_RESPONSE 0x80
/*
* All operation messages (both requests and responses) begin with
* a common header that encodes the size of the data (header
......@@ -61,7 +67,7 @@ int gb_operation_wait(struct gb_operation *operation)
ret = wait_for_completion_interruptible(&operation->completion);
/* If interrupted, cancel the in-flight buffer */
if (ret < 0)
ret = greybus_kill_gbuf(operation->gbuf);
ret = greybus_kill_gbuf(operation->request);
return ret;
}
......@@ -80,12 +86,14 @@ int gb_operation_submit(struct gb_operation *operation,
{
int ret;
/* XXX
* gfp is probably GFP_ATOMIC but really I think
* the gfp mask should go away.
/*
* XXX
* I think the order of operations is going to be
* significant, and if so, we may need a mutex to surround
* setting the operation id and submitting the gbuf.
*/
operation->callback = callback;
ret = greybus_submit_gbuf(operation->gbuf, GFP_KERNEL);
ret = greybus_submit_gbuf(operation->request, GFP_KERNEL);
if (ret)
return ret;
if (!callback)
......@@ -95,52 +103,100 @@ int gb_operation_submit(struct gb_operation *operation,
}
/*
* Called when a greybus request message has actually been sent.
* Called when an operation buffer completes.
*/
static void gbuf_out_callback(struct gbuf *gbuf)
static void gb_operation_gbuf_complete(struct gbuf *gbuf)
{
/* Record it's been submitted; need response now */
/* TODO */
}
/*
* Allocate a buffer to be used for an operation request or response
* message. Both types of message contain a header, which is filled
* in here. W
*/
struct gbuf *gb_operation_gbuf_create(struct gb_operation *operation,
u8 type, size_t size, bool outbound)
{
struct gb_connection *connection = operation->connection;
struct gb_operation_msg_hdr *header;
struct gbuf *gbuf;
gfp_t gfp_flags = outbound ? GFP_KERNEL : GFP_ATOMIC;
/* Operation buffers hold a header in addition to their payload */
size += sizeof(*header);
gbuf = greybus_alloc_gbuf(connection, gb_operation_gbuf_complete,
size, outbound, gfp_flags, operation);
if (!gbuf)
return NULL;
/* Fill in the header structure */
header = (struct gb_operation_msg_hdr *)gbuf->transfer_buffer;
header->size = cpu_to_le16(size);
header->id = 0; /* Filled in when submitted */
header->type = type;
return gbuf;
}
/*
* Create a Greybus operation having a buffer big enough for an
* outgoing payload of the given size to be sent over the given
* connection.
* Create a Greybus operation to be sent over the given connection.
* The request buffer will big enough for a payload of the given
* size. Outgoing requests must specify the size of the response
* buffer size, which must be sufficient to hold all expected
* response data.
*
* Returns a pointer to the new operation or a null pointer if a
* failure occurs due to memory exhaustion.
* Incoming requests will supply a response size of 0, and in that
* case no response buffer is allocated. (A response always
* includes a status byte, so 0 is not a valid size.) Whatever
* handles the operation request is responsible for allocating the
* response buffer.
*
* Returns a pointer to the new operation or a null pointer if an
* error occurs.
*/
struct gb_operation *gb_operation_create(struct gb_connection *connection,
size_t size, u8 type)
u8 type, size_t request_size,
size_t response_size)
{
struct gb_operation *operation;
struct gb_operation_msg_hdr *header;
struct gbuf *gbuf;
gfp_t gfp_flags = response_size ? GFP_KERNEL : GFP_ATOMIC;
if (!request_size) {
gb_connection_err(connection, "zero-sized request");
return NULL;
}
/* XXX Use a slab cache */
operation = kzalloc(sizeof(*operation), GFP_KERNEL);
operation = kzalloc(sizeof(*operation), gfp_flags);
if (!operation)
return NULL;
operation->connection = connection; /* XXX refcount? */
/* Our buffer holds a header in addition to the requested payload */
size += sizeof(*header);
gbuf = greybus_alloc_gbuf(connection, gbuf_out_callback, size,
true, GFP_KERNEL, operation);
if (gbuf) {
operation->request = gb_operation_gbuf_create(operation, type,
request_size, true);
if (!operation->request) {
kfree(operation);
return NULL;
}
gbuf->actual_length = size; /* Record what we'll use */
operation->connection = connection; /* XXX refcount? */
/* Fill in the header structure and payload pointer */
operation->gbuf = gbuf;
header = (struct gb_operation_msg_hdr *)&gbuf->transfer_buffer;
header->size = cpu_to_le16(size);
header->id = 0; /* Filled in when submitted */
header->type = type;
operation->payload = (char *)header + sizeof(*header);
operation->request_payload = operation->request->transfer_buffer +
sizeof(struct gb_operation_msg_hdr);
/* We always use the full request buffer */
operation->request->actual_length = request_size;
if (response_size) {
type |= GB_OPERATION_TYPE_RESPONSE;
operation->response = gb_operation_gbuf_create(operation,
type, response_size, false);
if (!operation->response) {
greybus_free_gbuf(operation->request);
kfree(operation);
return NULL;
}
operation->response_payload =
operation->response->transfer_buffer +
sizeof(struct gb_operation_msg_hdr);
}
operation->callback = NULL; /* set at submit time */
init_completion(&operation->completion);
......@@ -165,7 +221,8 @@ void gb_operation_destroy(struct gb_operation *operation)
list_del(&operation->links);
spin_unlock_irq(&gb_operations_lock);
greybus_free_gbuf(operation->gbuf);
greybus_free_gbuf(operation->response);
greybus_free_gbuf(operation->request);
kfree(operation);
}
......@@ -50,17 +50,23 @@ struct gb_operation;
typedef void (*gb_operation_callback)(struct gb_operation *);
struct gb_operation {
struct gb_connection *connection;
struct gbuf *gbuf;
void *payload; /* sender data */
struct gbuf *request;
struct gbuf *response;
gb_operation_callback callback; /* If asynchronous */
struct completion completion; /* Used if no callback */
u8 result;
struct list_head links; /* connection->operations */
/* These are what's used by caller */
void *request_payload;
void *response_payload;
};
struct gb_operation *gb_operation_create(struct gb_connection *connection,
size_t size, u8 type);
u8 type, size_t request_size,
size_t response_size);
void gb_operation_destroy(struct gb_operation *operation);
int gb_operation_wait(struct gb_operation *operation);
......
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