Julian Hall | 2dcd69c | 2020-11-23 18:05:04 +0100 | [diff] [blame] | 1 | /* |
julhal01 | c3f4e9a | 2020-12-15 13:39:01 +0000 | [diff] [blame] | 2 | * Copyright (c) 2020-2021, Arm Limited and Contributors. All rights reserved. |
Julian Hall | 2dcd69c | 2020-11-23 18:05:04 +0100 | [diff] [blame] | 3 | * |
| 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> |
| 10 | |
| 11 | static const struct service_handler *find_handler(const struct service_provider *sp, |
julhal01 | c3f4e9a | 2020-12-15 13:39:01 +0000 | [diff] [blame] | 12 | uint32_t opcode) |
Julian Hall | 2dcd69c | 2020-11-23 18:05:04 +0100 | [diff] [blame] | 13 | { |
| 14 | const struct service_handler *handler = NULL; |
Julian Hall | 2dcd69c | 2020-11-23 18:05:04 +0100 | [diff] [blame] | 15 | |
Julian Hall | 65ff905 | 2021-07-27 09:08:32 +0100 | [diff] [blame^] | 16 | if ((opcode >= sp->opcode_range_lo) && (opcode <= sp->opcode_range_hi)) { |
| 17 | |
| 18 | size_t index = 0; |
| 19 | |
Julian Hall | 2dcd69c | 2020-11-23 18:05:04 +0100 | [diff] [blame] | 20 | 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 Hall | 65ff905 | 2021-07-27 09:08:32 +0100 | [diff] [blame^] | 32 | static 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++) { |
| 43 | |
| 44 | uint32_t opcode = service_handler_get_opcode(&sp->handlers[index]); |
| 45 | |
| 46 | if (opcode < lo) lo = opcode; |
| 47 | if (opcode > hi) hi = opcode; |
| 48 | } |
| 49 | |
| 50 | sp->opcode_range_lo = lo; |
| 51 | sp->opcode_range_hi = hi; |
| 52 | } |
| 53 | |
julhal01 | c3f4e9a | 2020-12-15 13:39:01 +0000 | [diff] [blame] | 54 | static rpc_status_t receive(struct rpc_interface *rpc_iface, struct call_req *req) |
Julian Hall | 2dcd69c | 2020-11-23 18:05:04 +0100 | [diff] [blame] | 55 | { |
| 56 | rpc_status_t rpc_status; |
| 57 | struct service_provider *sp = NULL; |
| 58 | const struct service_handler *handler = NULL; |
| 59 | |
julhal01 | c3f4e9a | 2020-12-15 13:39:01 +0000 | [diff] [blame] | 60 | sp = (struct service_provider*)((char*)rpc_iface - offsetof(struct service_provider, iface)); |
Julian Hall | 2dcd69c | 2020-11-23 18:05:04 +0100 | [diff] [blame] | 61 | handler = find_handler(sp, call_req_get_opcode(req)); |
| 62 | |
julhal01 | c3f4e9a | 2020-12-15 13:39:01 +0000 | [diff] [blame] | 63 | if (handler) { |
Julian Hall | 2dcd69c | 2020-11-23 18:05:04 +0100 | [diff] [blame] | 64 | |
julhal01 | c3f4e9a | 2020-12-15 13:39:01 +0000 | [diff] [blame] | 65 | rpc_status = service_handler_invoke(handler, rpc_iface->context, req); |
| 66 | } |
| 67 | else if (sp->successor) { |
Julian Hall | 2dcd69c | 2020-11-23 18:05:04 +0100 | [diff] [blame] | 68 | |
julhal01 | c3f4e9a | 2020-12-15 13:39:01 +0000 | [diff] [blame] | 69 | rpc_status = rpc_interface_receive(sp->successor, req); |
| 70 | } |
| 71 | else { |
| 72 | |
| 73 | rpc_status = TS_RPC_ERROR_INVALID_OPCODE; |
| 74 | } |
Julian Hall | 2dcd69c | 2020-11-23 18:05:04 +0100 | [diff] [blame] | 75 | |
| 76 | return rpc_status; |
| 77 | } |
| 78 | |
| 79 | void service_provider_init(struct service_provider *sp, void *context, |
julhal01 | c3f4e9a | 2020-12-15 13:39:01 +0000 | [diff] [blame] | 80 | const struct service_handler *handlers, |
| 81 | size_t num_handlers) |
Julian Hall | 2dcd69c | 2020-11-23 18:05:04 +0100 | [diff] [blame] | 82 | { |
julhal01 | c3f4e9a | 2020-12-15 13:39:01 +0000 | [diff] [blame] | 83 | sp->iface.receive = receive; |
| 84 | sp->iface.context = context; |
Julian Hall | 2dcd69c | 2020-11-23 18:05:04 +0100 | [diff] [blame] | 85 | |
| 86 | sp->handlers = handlers; |
| 87 | sp->num_handlers = num_handlers; |
| 88 | |
julhal01 | c3f4e9a | 2020-12-15 13:39:01 +0000 | [diff] [blame] | 89 | sp->successor = NULL; |
Julian Hall | 65ff905 | 2021-07-27 09:08:32 +0100 | [diff] [blame^] | 90 | |
| 91 | set_opcode_range(sp); |
Julian Hall | 2dcd69c | 2020-11-23 18:05:04 +0100 | [diff] [blame] | 92 | } |
Julian Hall | 13e7695 | 2021-07-13 12:17:09 +0100 | [diff] [blame] | 93 | |
| 94 | void service_provider_extend(struct service_provider *context, |
| 95 | struct service_provider *sub_provider) |
| 96 | { |
| 97 | sub_provider->successor = context->successor; |
| 98 | context->successor = &sub_provider->iface; |
| 99 | } |