blob: b6fbe3d83bd039b526afb00a08c2f6b7f4797a25 [file] [log] [blame]
Julian Hall2dcd69c2020-11-23 18:05:04 +01001/*
julhal01c3f4e9a2020-12-15 13:39:01 +00002 * Copyright (c) 2020-2021, Arm Limited and Contributors. All rights reserved.
Julian Hall2dcd69c2020-11-23 18:05:04 +01003 *
4 * SPDX-License-Identifier: BSD-3-Clause
5 */
6
7#include "service_provider.h"
8#include <protocols/rpc/common/packed-c/status.h>
9#include <stddef.h>
Imre Kis82ba61b2023-07-04 13:53:39 +020010#include <string.h>
Julian Hall2dcd69c2020-11-23 18:05:04 +010011
12static const struct service_handler *find_handler(const struct service_provider *sp,
Imre Kis82ba61b2023-07-04 13:53:39 +020013 uint32_t opcode)
Julian Hall2dcd69c2020-11-23 18:05:04 +010014{
15 const struct service_handler *handler = NULL;
Julian Hall2dcd69c2020-11-23 18:05:04 +010016
Julian Hall65ff9052021-07-27 09:08:32 +010017 if ((opcode >= sp->opcode_range_lo) && (opcode <= sp->opcode_range_hi)) {
Julian Hall65ff9052021-07-27 09:08:32 +010018 size_t index = 0;
19
Julian Hall2dcd69c2020-11-23 18:05:04 +010020 while (index < sp->num_handlers) {
21 if (service_handler_get_opcode(&sp->handlers[index]) == opcode) {
22 handler = &sp->handlers[index];
23 break;
24 }
25 ++index;
26 }
27 }
28
29 return handler;
30}
31
Julian Hall65ff9052021-07-27 09:08:32 +010032static void set_opcode_range(struct service_provider *sp)
33{
34 uint32_t lo = UINT32_MAX;
35 uint32_t hi = 0;
36
37 /* Determine opcode range so that this service may be skipped
38 * without having to iterate over all handlers. This reduces
39 * the time to find a qualifying handler when multiple service
40 * providers are chained.
41 */
42 for (size_t index = 0; index < sp->num_handlers; index++) {
Julian Hall65ff9052021-07-27 09:08:32 +010043 uint32_t opcode = service_handler_get_opcode(&sp->handlers[index]);
44
45 if (opcode < lo) lo = opcode;
46 if (opcode > hi) hi = opcode;
47 }
48
49 sp->opcode_range_lo = lo;
50 sp->opcode_range_hi = hi;
51}
52
Imre Kis82ba61b2023-07-04 13:53:39 +020053static rpc_status_t receive(void *context, struct rpc_request *req)
Julian Hall2dcd69c2020-11-23 18:05:04 +010054{
Imre Kis82ba61b2023-07-04 13:53:39 +020055 struct rpc_service_interface *rpc_iface = (struct rpc_service_interface *)context;
Julian Hall2dcd69c2020-11-23 18:05:04 +010056 rpc_status_t rpc_status;
57 struct service_provider *sp = NULL;
58 const struct service_handler *handler = NULL;
59
julhal01c3f4e9a2020-12-15 13:39:01 +000060 sp = (struct service_provider*)((char*)rpc_iface - offsetof(struct service_provider, iface));
Imre Kis82ba61b2023-07-04 13:53:39 +020061 handler = find_handler(sp, req->opcode);
Julian Hall2dcd69c2020-11-23 18:05:04 +010062
julhal01c3f4e9a2020-12-15 13:39:01 +000063 if (handler) {
Imre Kis82ba61b2023-07-04 13:53:39 +020064 rpc_status = service_handler_invoke(handler, rpc_iface->context, req);
65 } else if (sp->successor) {
66 rpc_status = rpc_service_receive(sp->successor, req);
67 } else {
68 rpc_status = RPC_ERROR_INVALID_VALUE;
julhal01c3f4e9a2020-12-15 13:39:01 +000069 }
Julian Hall2dcd69c2020-11-23 18:05:04 +010070
71 return rpc_status;
72}
73
74void service_provider_init(struct service_provider *sp, void *context,
Imre Kis82ba61b2023-07-04 13:53:39 +020075 const struct rpc_uuid *service_uuid,
76 const struct service_handler *handlers,
77 size_t num_handlers)
Julian Hall2dcd69c2020-11-23 18:05:04 +010078{
julhal01c3f4e9a2020-12-15 13:39:01 +000079 sp->iface.receive = receive;
80 sp->iface.context = context;
Imre Kis82ba61b2023-07-04 13:53:39 +020081 memcpy(&sp->iface.uuid, service_uuid, sizeof(sp->iface.uuid));
Julian Hall2dcd69c2020-11-23 18:05:04 +010082
83 sp->handlers = handlers;
84 sp->num_handlers = num_handlers;
85
julhal01c3f4e9a2020-12-15 13:39:01 +000086 sp->successor = NULL;
Julian Hall65ff9052021-07-27 09:08:32 +010087
88 set_opcode_range(sp);
Julian Hall2dcd69c2020-11-23 18:05:04 +010089}
Julian Hall13e76952021-07-13 12:17:09 +010090
91void service_provider_extend(struct service_provider *context,
Imre Kis82ba61b2023-07-04 13:53:39 +020092 struct service_provider *sub_provider)
Julian Hall13e76952021-07-13 12:17:09 +010093{
94 sub_provider->successor = context->successor;
95 context->successor = &sub_provider->iface;
96}