]>
Commit | Line | Data |
---|---|---|
9b8a11e8 BA |
1 | // SPDX-License-Identifier: GPL-2.0 |
2 | /* | |
3 | * Copyright (c) 2012-2014, The Linux Foundation. All rights reserved. | |
4 | * Copyright (c) 2017, Linaro Ltd. | |
5 | */ | |
6 | #ifndef __QMI_HELPERS_H__ | |
7 | #define __QMI_HELPERS_H__ | |
8 | ||
3830d077 BA |
9 | #include <linux/completion.h> |
10 | #include <linux/idr.h> | |
11 | #include <linux/list.h> | |
12 | #include <linux/qrtr.h> | |
9b8a11e8 | 13 | #include <linux/types.h> |
3830d077 BA |
14 | #include <linux/workqueue.h> |
15 | ||
16 | struct socket; | |
9b8a11e8 BA |
17 | |
18 | /** | |
19 | * qmi_header - wireformat header of QMI messages | |
20 | * @type: type of message | |
21 | * @txn_id: transaction id | |
22 | * @msg_id: message id | |
23 | * @msg_len: length of message payload following header | |
24 | */ | |
25 | struct qmi_header { | |
26 | u8 type; | |
27 | u16 txn_id; | |
28 | u16 msg_id; | |
29 | u16 msg_len; | |
30 | } __packed; | |
31 | ||
32 | #define QMI_REQUEST 0 | |
33 | #define QMI_RESPONSE 2 | |
34 | #define QMI_INDICATION 4 | |
35 | ||
36 | #define QMI_COMMON_TLV_TYPE 0 | |
37 | ||
38 | enum qmi_elem_type { | |
39 | QMI_EOTI, | |
40 | QMI_OPT_FLAG, | |
41 | QMI_DATA_LEN, | |
42 | QMI_UNSIGNED_1_BYTE, | |
43 | QMI_UNSIGNED_2_BYTE, | |
44 | QMI_UNSIGNED_4_BYTE, | |
45 | QMI_UNSIGNED_8_BYTE, | |
46 | QMI_SIGNED_2_BYTE_ENUM, | |
47 | QMI_SIGNED_4_BYTE_ENUM, | |
48 | QMI_STRUCT, | |
49 | QMI_STRING, | |
50 | }; | |
51 | ||
52 | enum qmi_array_type { | |
53 | NO_ARRAY, | |
54 | STATIC_ARRAY, | |
55 | VAR_LEN_ARRAY, | |
56 | }; | |
57 | ||
58 | /** | |
59 | * struct qmi_elem_info - describes how to encode a single QMI element | |
60 | * @data_type: Data type of this element. | |
61 | * @elem_len: Array length of this element, if an array. | |
62 | * @elem_size: Size of a single instance of this data type. | |
63 | * @array_type: Array type of this element. | |
64 | * @tlv_type: QMI message specific type to identify which element | |
65 | * is present in an incoming message. | |
66 | * @offset: Specifies the offset of the first instance of this | |
67 | * element in the data structure. | |
68 | * @ei_array: Null-terminated array of @qmi_elem_info to describe nested | |
69 | * structures. | |
70 | */ | |
71 | struct qmi_elem_info { | |
72 | enum qmi_elem_type data_type; | |
73 | u32 elem_len; | |
74 | u32 elem_size; | |
75 | enum qmi_array_type array_type; | |
76 | u8 tlv_type; | |
77 | u32 offset; | |
78 | struct qmi_elem_info *ei_array; | |
79 | }; | |
80 | ||
81 | #define QMI_RESULT_SUCCESS_V01 0 | |
82 | #define QMI_RESULT_FAILURE_V01 1 | |
83 | ||
84 | #define QMI_ERR_NONE_V01 0 | |
85 | #define QMI_ERR_MALFORMED_MSG_V01 1 | |
86 | #define QMI_ERR_NO_MEMORY_V01 2 | |
87 | #define QMI_ERR_INTERNAL_V01 3 | |
88 | #define QMI_ERR_CLIENT_IDS_EXHAUSTED_V01 5 | |
89 | #define QMI_ERR_INVALID_ID_V01 41 | |
90 | #define QMI_ERR_ENCODING_V01 58 | |
91 | #define QMI_ERR_INCOMPATIBLE_STATE_V01 90 | |
92 | #define QMI_ERR_NOT_SUPPORTED_V01 94 | |
93 | ||
94 | /** | |
95 | * qmi_response_type_v01 - common response header (decoded) | |
96 | * @result: result of the transaction | |
97 | * @error: error value, when @result is QMI_RESULT_FAILURE_V01 | |
98 | */ | |
99 | struct qmi_response_type_v01 { | |
100 | u16 result; | |
101 | u16 error; | |
102 | }; | |
103 | ||
104 | extern struct qmi_elem_info qmi_response_type_v01_ei[]; | |
105 | ||
3830d077 BA |
106 | /** |
107 | * struct qmi_service - context to track lookup-results | |
108 | * @service: service type | |
109 | * @version: version of the @service | |
110 | * @instance: instance id of the @service | |
111 | * @node: node of the service | |
112 | * @port: port of the service | |
113 | * @priv: handle for client's use | |
114 | * @list_node: list_head for house keeping | |
115 | */ | |
116 | struct qmi_service { | |
117 | unsigned int service; | |
118 | unsigned int version; | |
119 | unsigned int instance; | |
120 | ||
121 | unsigned int node; | |
122 | unsigned int port; | |
123 | ||
124 | void *priv; | |
125 | struct list_head list_node; | |
126 | }; | |
127 | ||
128 | struct qmi_handle; | |
129 | ||
130 | /** | |
131 | * struct qmi_ops - callbacks for qmi_handle | |
132 | * @new_server: inform client of a new_server lookup-result, returning | |
133 | * successfully from this call causes the library to call | |
134 | * @del_server as the service is removed from the | |
135 | * lookup-result. @priv of the qmi_service can be used by | |
136 | * the client | |
137 | * @del_server: inform client of a del_server lookup-result | |
138 | * @net_reset: inform client that the name service was restarted and | |
139 | * that and any state needs to be released | |
140 | * @msg_handler: invoked for incoming messages, allows a client to | |
141 | * override the usual QMI message handler | |
142 | * @bye: inform a client that all clients from a node are gone | |
143 | * @del_client: inform a client that a particular client is gone | |
144 | */ | |
145 | struct qmi_ops { | |
146 | int (*new_server)(struct qmi_handle *qmi, struct qmi_service *svc); | |
147 | void (*del_server)(struct qmi_handle *qmi, struct qmi_service *svc); | |
148 | void (*net_reset)(struct qmi_handle *qmi); | |
149 | void (*msg_handler)(struct qmi_handle *qmi, struct sockaddr_qrtr *sq, | |
150 | const void *data, size_t count); | |
151 | void (*bye)(struct qmi_handle *qmi, unsigned int node); | |
152 | void (*del_client)(struct qmi_handle *qmi, | |
153 | unsigned int node, unsigned int port); | |
154 | }; | |
155 | ||
156 | /** | |
157 | * struct qmi_txn - transaction context | |
158 | * @qmi: QMI handle this transaction is associated with | |
159 | * @id: transaction id | |
160 | * @lock: for synchronization between handler and waiter of messages | |
161 | * @completion: completion object as the transaction receives a response | |
162 | * @result: result code for the completed transaction | |
163 | * @ei: description of the QMI encoded response (optional) | |
164 | * @dest: destination buffer to decode message into (optional) | |
165 | */ | |
166 | struct qmi_txn { | |
167 | struct qmi_handle *qmi; | |
168 | ||
c4fe17e0 | 169 | u16 id; |
3830d077 BA |
170 | |
171 | struct mutex lock; | |
172 | struct completion completion; | |
173 | int result; | |
174 | ||
175 | struct qmi_elem_info *ei; | |
176 | void *dest; | |
177 | }; | |
178 | ||
179 | /** | |
180 | * struct qmi_msg_handler - description of QMI message handler | |
181 | * @type: type of message | |
182 | * @msg_id: message id | |
183 | * @ei: description of the QMI encoded message | |
184 | * @decoded_size: size of the decoded object | |
185 | * @fn: function to invoke as the message is decoded | |
186 | */ | |
187 | struct qmi_msg_handler { | |
188 | unsigned int type; | |
189 | unsigned int msg_id; | |
190 | ||
191 | struct qmi_elem_info *ei; | |
192 | ||
193 | size_t decoded_size; | |
194 | void (*fn)(struct qmi_handle *qmi, struct sockaddr_qrtr *sq, | |
195 | struct qmi_txn *txn, const void *decoded); | |
196 | }; | |
197 | ||
198 | /** | |
199 | * struct qmi_handle - QMI context | |
200 | * @sock: socket handle | |
201 | * @sock_lock: synchronization of @sock modifications | |
202 | * @sq: sockaddr of @sock | |
203 | * @work: work for handling incoming messages | |
204 | * @wq: workqueue to post @work on | |
205 | * @recv_buf: scratch buffer for handling incoming messages | |
206 | * @recv_buf_size: size of @recv_buf | |
207 | * @lookups: list of registered lookup requests | |
208 | * @lookup_results: list of lookup-results advertised to the client | |
209 | * @services: list of registered services (by this client) | |
210 | * @ops: reference to callbacks | |
211 | * @txns: outstanding transactions | |
212 | * @txn_lock: lock for modifications of @txns | |
213 | * @handlers: list of handlers for incoming messages | |
214 | */ | |
215 | struct qmi_handle { | |
216 | struct socket *sock; | |
217 | struct mutex sock_lock; | |
218 | ||
219 | struct sockaddr_qrtr sq; | |
220 | ||
221 | struct work_struct work; | |
222 | struct workqueue_struct *wq; | |
223 | ||
224 | void *recv_buf; | |
225 | size_t recv_buf_size; | |
226 | ||
227 | struct list_head lookups; | |
228 | struct list_head lookup_results; | |
229 | struct list_head services; | |
230 | ||
231 | struct qmi_ops ops; | |
232 | ||
233 | struct idr txns; | |
234 | struct mutex txn_lock; | |
235 | ||
236 | const struct qmi_msg_handler *handlers; | |
237 | }; | |
238 | ||
239 | int qmi_add_lookup(struct qmi_handle *qmi, unsigned int service, | |
240 | unsigned int version, unsigned int instance); | |
241 | int qmi_add_server(struct qmi_handle *qmi, unsigned int service, | |
242 | unsigned int version, unsigned int instance); | |
243 | ||
244 | int qmi_handle_init(struct qmi_handle *qmi, size_t max_msg_len, | |
245 | const struct qmi_ops *ops, | |
246 | const struct qmi_msg_handler *handlers); | |
247 | void qmi_handle_release(struct qmi_handle *qmi); | |
248 | ||
249 | ssize_t qmi_send_request(struct qmi_handle *qmi, struct sockaddr_qrtr *sq, | |
250 | struct qmi_txn *txn, int msg_id, size_t len, | |
251 | struct qmi_elem_info *ei, const void *c_struct); | |
252 | ssize_t qmi_send_response(struct qmi_handle *qmi, struct sockaddr_qrtr *sq, | |
253 | struct qmi_txn *txn, int msg_id, size_t len, | |
254 | struct qmi_elem_info *ei, const void *c_struct); | |
255 | ssize_t qmi_send_indication(struct qmi_handle *qmi, struct sockaddr_qrtr *sq, | |
256 | int msg_id, size_t len, struct qmi_elem_info *ei, | |
257 | const void *c_struct); | |
258 | ||
9b8a11e8 BA |
259 | void *qmi_encode_message(int type, unsigned int msg_id, size_t *len, |
260 | unsigned int txn_id, struct qmi_elem_info *ei, | |
261 | const void *c_struct); | |
262 | ||
263 | int qmi_decode_message(const void *buf, size_t len, | |
264 | struct qmi_elem_info *ei, void *c_struct); | |
265 | ||
3830d077 BA |
266 | int qmi_txn_init(struct qmi_handle *qmi, struct qmi_txn *txn, |
267 | struct qmi_elem_info *ei, void *c_struct); | |
268 | int qmi_txn_wait(struct qmi_txn *txn, unsigned long timeout); | |
269 | void qmi_txn_cancel(struct qmi_txn *txn); | |
270 | ||
9b8a11e8 | 271 | #endif |