Loading...
1// SPDX-License-Identifier: GPL-2.0
2/*
3 * System Control and Management Interface (SCMI) Sensor Protocol
4 *
5 * Copyright (C) 2018 ARM Ltd.
6 */
7
8#define pr_fmt(fmt) "SCMI Notifications SENSOR - " fmt
9
10#include <linux/scmi_protocol.h>
11
12#include "common.h"
13#include "notify.h"
14
15enum scmi_sensor_protocol_cmd {
16 SENSOR_DESCRIPTION_GET = 0x3,
17 SENSOR_TRIP_POINT_NOTIFY = 0x4,
18 SENSOR_TRIP_POINT_CONFIG = 0x5,
19 SENSOR_READING_GET = 0x6,
20};
21
22struct scmi_msg_resp_sensor_attributes {
23 __le16 num_sensors;
24 u8 max_requests;
25 u8 reserved;
26 __le32 reg_addr_low;
27 __le32 reg_addr_high;
28 __le32 reg_size;
29};
30
31struct scmi_msg_resp_sensor_description {
32 __le16 num_returned;
33 __le16 num_remaining;
34 struct {
35 __le32 id;
36 __le32 attributes_low;
37#define SUPPORTS_ASYNC_READ(x) ((x) & BIT(31))
38#define NUM_TRIP_POINTS(x) ((x) & 0xff)
39 __le32 attributes_high;
40#define SENSOR_TYPE(x) ((x) & 0xff)
41#define SENSOR_SCALE(x) (((x) >> 11) & 0x1f)
42#define SENSOR_SCALE_SIGN BIT(4)
43#define SENSOR_SCALE_EXTEND GENMASK(7, 5)
44#define SENSOR_UPDATE_SCALE(x) (((x) >> 22) & 0x1f)
45#define SENSOR_UPDATE_BASE(x) (((x) >> 27) & 0x1f)
46 u8 name[SCMI_MAX_STR_SIZE];
47 } desc[0];
48};
49
50struct scmi_msg_sensor_trip_point_notify {
51 __le32 id;
52 __le32 event_control;
53#define SENSOR_TP_NOTIFY_ALL BIT(0)
54};
55
56struct scmi_msg_set_sensor_trip_point {
57 __le32 id;
58 __le32 event_control;
59#define SENSOR_TP_EVENT_MASK (0x3)
60#define SENSOR_TP_DISABLED 0x0
61#define SENSOR_TP_POSITIVE 0x1
62#define SENSOR_TP_NEGATIVE 0x2
63#define SENSOR_TP_BOTH 0x3
64#define SENSOR_TP_ID(x) (((x) & 0xff) << 4)
65 __le32 value_low;
66 __le32 value_high;
67};
68
69struct scmi_msg_sensor_reading_get {
70 __le32 id;
71 __le32 flags;
72#define SENSOR_READ_ASYNC BIT(0)
73};
74
75struct scmi_sensor_trip_notify_payld {
76 __le32 agent_id;
77 __le32 sensor_id;
78 __le32 trip_point_desc;
79};
80
81struct sensors_info {
82 u32 version;
83 int num_sensors;
84 int max_requests;
85 u64 reg_addr;
86 u32 reg_size;
87 struct scmi_sensor_info *sensors;
88};
89
90static int scmi_sensor_attributes_get(const struct scmi_handle *handle,
91 struct sensors_info *si)
92{
93 int ret;
94 struct scmi_xfer *t;
95 struct scmi_msg_resp_sensor_attributes *attr;
96
97 ret = scmi_xfer_get_init(handle, PROTOCOL_ATTRIBUTES,
98 SCMI_PROTOCOL_SENSOR, 0, sizeof(*attr), &t);
99 if (ret)
100 return ret;
101
102 attr = t->rx.buf;
103
104 ret = scmi_do_xfer(handle, t);
105 if (!ret) {
106 si->num_sensors = le16_to_cpu(attr->num_sensors);
107 si->max_requests = attr->max_requests;
108 si->reg_addr = le32_to_cpu(attr->reg_addr_low) |
109 (u64)le32_to_cpu(attr->reg_addr_high) << 32;
110 si->reg_size = le32_to_cpu(attr->reg_size);
111 }
112
113 scmi_xfer_put(handle, t);
114 return ret;
115}
116
117static int scmi_sensor_description_get(const struct scmi_handle *handle,
118 struct sensors_info *si)
119{
120 int ret, cnt;
121 u32 desc_index = 0;
122 u16 num_returned, num_remaining;
123 struct scmi_xfer *t;
124 struct scmi_msg_resp_sensor_description *buf;
125
126 ret = scmi_xfer_get_init(handle, SENSOR_DESCRIPTION_GET,
127 SCMI_PROTOCOL_SENSOR, sizeof(__le32), 0, &t);
128 if (ret)
129 return ret;
130
131 buf = t->rx.buf;
132
133 do {
134 /* Set the number of sensors to be skipped/already read */
135 put_unaligned_le32(desc_index, t->tx.buf);
136
137 ret = scmi_do_xfer(handle, t);
138 if (ret)
139 break;
140
141 num_returned = le16_to_cpu(buf->num_returned);
142 num_remaining = le16_to_cpu(buf->num_remaining);
143
144 if (desc_index + num_returned > si->num_sensors) {
145 dev_err(handle->dev, "No. of sensors can't exceed %d",
146 si->num_sensors);
147 break;
148 }
149
150 for (cnt = 0; cnt < num_returned; cnt++) {
151 u32 attrh, attrl;
152 struct scmi_sensor_info *s;
153
154 attrl = le32_to_cpu(buf->desc[cnt].attributes_low);
155 attrh = le32_to_cpu(buf->desc[cnt].attributes_high);
156 s = &si->sensors[desc_index + cnt];
157 s->id = le32_to_cpu(buf->desc[cnt].id);
158 s->type = SENSOR_TYPE(attrh);
159 s->scale = SENSOR_SCALE(attrh);
160 /* Sign extend to a full s8 */
161 if (s->scale & SENSOR_SCALE_SIGN)
162 s->scale |= SENSOR_SCALE_EXTEND;
163 s->async = SUPPORTS_ASYNC_READ(attrl);
164 s->num_trip_points = NUM_TRIP_POINTS(attrl);
165 strlcpy(s->name, buf->desc[cnt].name, SCMI_MAX_STR_SIZE);
166 }
167
168 desc_index += num_returned;
169 /*
170 * check for both returned and remaining to avoid infinite
171 * loop due to buggy firmware
172 */
173 } while (num_returned && num_remaining);
174
175 scmi_xfer_put(handle, t);
176 return ret;
177}
178
179static int scmi_sensor_trip_point_notify(const struct scmi_handle *handle,
180 u32 sensor_id, bool enable)
181{
182 int ret;
183 u32 evt_cntl = enable ? SENSOR_TP_NOTIFY_ALL : 0;
184 struct scmi_xfer *t;
185 struct scmi_msg_sensor_trip_point_notify *cfg;
186
187 ret = scmi_xfer_get_init(handle, SENSOR_TRIP_POINT_NOTIFY,
188 SCMI_PROTOCOL_SENSOR, sizeof(*cfg), 0, &t);
189 if (ret)
190 return ret;
191
192 cfg = t->tx.buf;
193 cfg->id = cpu_to_le32(sensor_id);
194 cfg->event_control = cpu_to_le32(evt_cntl);
195
196 ret = scmi_do_xfer(handle, t);
197
198 scmi_xfer_put(handle, t);
199 return ret;
200}
201
202static int
203scmi_sensor_trip_point_config(const struct scmi_handle *handle, u32 sensor_id,
204 u8 trip_id, u64 trip_value)
205{
206 int ret;
207 u32 evt_cntl = SENSOR_TP_BOTH;
208 struct scmi_xfer *t;
209 struct scmi_msg_set_sensor_trip_point *trip;
210
211 ret = scmi_xfer_get_init(handle, SENSOR_TRIP_POINT_CONFIG,
212 SCMI_PROTOCOL_SENSOR, sizeof(*trip), 0, &t);
213 if (ret)
214 return ret;
215
216 trip = t->tx.buf;
217 trip->id = cpu_to_le32(sensor_id);
218 trip->event_control = cpu_to_le32(evt_cntl | SENSOR_TP_ID(trip_id));
219 trip->value_low = cpu_to_le32(trip_value & 0xffffffff);
220 trip->value_high = cpu_to_le32(trip_value >> 32);
221
222 ret = scmi_do_xfer(handle, t);
223
224 scmi_xfer_put(handle, t);
225 return ret;
226}
227
228static int scmi_sensor_reading_get(const struct scmi_handle *handle,
229 u32 sensor_id, u64 *value)
230{
231 int ret;
232 struct scmi_xfer *t;
233 struct scmi_msg_sensor_reading_get *sensor;
234 struct sensors_info *si = handle->sensor_priv;
235 struct scmi_sensor_info *s = si->sensors + sensor_id;
236
237 ret = scmi_xfer_get_init(handle, SENSOR_READING_GET,
238 SCMI_PROTOCOL_SENSOR, sizeof(*sensor),
239 sizeof(u64), &t);
240 if (ret)
241 return ret;
242
243 sensor = t->tx.buf;
244 sensor->id = cpu_to_le32(sensor_id);
245
246 if (s->async) {
247 sensor->flags = cpu_to_le32(SENSOR_READ_ASYNC);
248 ret = scmi_do_xfer_with_response(handle, t);
249 if (!ret)
250 *value = get_unaligned_le64((void *)
251 ((__le32 *)t->rx.buf + 1));
252 } else {
253 sensor->flags = cpu_to_le32(0);
254 ret = scmi_do_xfer(handle, t);
255 if (!ret)
256 *value = get_unaligned_le64(t->rx.buf);
257 }
258
259 scmi_xfer_put(handle, t);
260 return ret;
261}
262
263static const struct scmi_sensor_info *
264scmi_sensor_info_get(const struct scmi_handle *handle, u32 sensor_id)
265{
266 struct sensors_info *si = handle->sensor_priv;
267
268 return si->sensors + sensor_id;
269}
270
271static int scmi_sensor_count_get(const struct scmi_handle *handle)
272{
273 struct sensors_info *si = handle->sensor_priv;
274
275 return si->num_sensors;
276}
277
278static struct scmi_sensor_ops sensor_ops = {
279 .count_get = scmi_sensor_count_get,
280 .info_get = scmi_sensor_info_get,
281 .trip_point_config = scmi_sensor_trip_point_config,
282 .reading_get = scmi_sensor_reading_get,
283};
284
285static int scmi_sensor_set_notify_enabled(const struct scmi_handle *handle,
286 u8 evt_id, u32 src_id, bool enable)
287{
288 int ret;
289
290 ret = scmi_sensor_trip_point_notify(handle, src_id, enable);
291 if (ret)
292 pr_debug("FAIL_ENABLED - evt[%X] dom[%d] - ret:%d\n",
293 evt_id, src_id, ret);
294
295 return ret;
296}
297
298static void *scmi_sensor_fill_custom_report(const struct scmi_handle *handle,
299 u8 evt_id, ktime_t timestamp,
300 const void *payld, size_t payld_sz,
301 void *report, u32 *src_id)
302{
303 const struct scmi_sensor_trip_notify_payld *p = payld;
304 struct scmi_sensor_trip_point_report *r = report;
305
306 if (evt_id != SCMI_EVENT_SENSOR_TRIP_POINT_EVENT ||
307 sizeof(*p) != payld_sz)
308 return NULL;
309
310 r->timestamp = timestamp;
311 r->agent_id = le32_to_cpu(p->agent_id);
312 r->sensor_id = le32_to_cpu(p->sensor_id);
313 r->trip_point_desc = le32_to_cpu(p->trip_point_desc);
314 *src_id = r->sensor_id;
315
316 return r;
317}
318
319static const struct scmi_event sensor_events[] = {
320 {
321 .id = SCMI_EVENT_SENSOR_TRIP_POINT_EVENT,
322 .max_payld_sz = sizeof(struct scmi_sensor_trip_notify_payld),
323 .max_report_sz = sizeof(struct scmi_sensor_trip_point_report),
324 },
325};
326
327static const struct scmi_event_ops sensor_event_ops = {
328 .set_notify_enabled = scmi_sensor_set_notify_enabled,
329 .fill_custom_report = scmi_sensor_fill_custom_report,
330};
331
332static int scmi_sensors_protocol_init(struct scmi_handle *handle)
333{
334 u32 version;
335 struct sensors_info *sinfo;
336
337 scmi_version_get(handle, SCMI_PROTOCOL_SENSOR, &version);
338
339 dev_dbg(handle->dev, "Sensor Version %d.%d\n",
340 PROTOCOL_REV_MAJOR(version), PROTOCOL_REV_MINOR(version));
341
342 sinfo = devm_kzalloc(handle->dev, sizeof(*sinfo), GFP_KERNEL);
343 if (!sinfo)
344 return -ENOMEM;
345
346 scmi_sensor_attributes_get(handle, sinfo);
347
348 sinfo->sensors = devm_kcalloc(handle->dev, sinfo->num_sensors,
349 sizeof(*sinfo->sensors), GFP_KERNEL);
350 if (!sinfo->sensors)
351 return -ENOMEM;
352
353 scmi_sensor_description_get(handle, sinfo);
354
355 scmi_register_protocol_events(handle,
356 SCMI_PROTOCOL_SENSOR, SCMI_PROTO_QUEUE_SZ,
357 &sensor_event_ops, sensor_events,
358 ARRAY_SIZE(sensor_events),
359 sinfo->num_sensors);
360
361 sinfo->version = version;
362 handle->sensor_ops = &sensor_ops;
363 handle->sensor_priv = sinfo;
364
365 return 0;
366}
367
368static int __init scmi_sensors_init(void)
369{
370 return scmi_protocol_register(SCMI_PROTOCOL_SENSOR,
371 &scmi_sensors_protocol_init);
372}
373subsys_initcall(scmi_sensors_init);
1// SPDX-License-Identifier: GPL-2.0
2/*
3 * System Control and Management Interface (SCMI) Sensor Protocol
4 *
5 * Copyright (C) 2018 ARM Ltd.
6 */
7
8#include "common.h"
9
10enum scmi_sensor_protocol_cmd {
11 SENSOR_DESCRIPTION_GET = 0x3,
12 SENSOR_TRIP_POINT_NOTIFY = 0x4,
13 SENSOR_TRIP_POINT_CONFIG = 0x5,
14 SENSOR_READING_GET = 0x6,
15};
16
17struct scmi_msg_resp_sensor_attributes {
18 __le16 num_sensors;
19 u8 max_requests;
20 u8 reserved;
21 __le32 reg_addr_low;
22 __le32 reg_addr_high;
23 __le32 reg_size;
24};
25
26struct scmi_msg_resp_sensor_description {
27 __le16 num_returned;
28 __le16 num_remaining;
29 struct {
30 __le32 id;
31 __le32 attributes_low;
32#define SUPPORTS_ASYNC_READ(x) ((x) & BIT(31))
33#define NUM_TRIP_POINTS(x) ((x) & 0xff)
34 __le32 attributes_high;
35#define SENSOR_TYPE(x) ((x) & 0xff)
36#define SENSOR_SCALE(x) (((x) >> 11) & 0x1f)
37#define SENSOR_SCALE_SIGN BIT(4)
38#define SENSOR_SCALE_EXTEND GENMASK(7, 5)
39#define SENSOR_UPDATE_SCALE(x) (((x) >> 22) & 0x1f)
40#define SENSOR_UPDATE_BASE(x) (((x) >> 27) & 0x1f)
41 u8 name[SCMI_MAX_STR_SIZE];
42 } desc[0];
43};
44
45struct scmi_msg_sensor_trip_point_notify {
46 __le32 id;
47 __le32 event_control;
48#define SENSOR_TP_NOTIFY_ALL BIT(0)
49};
50
51struct scmi_msg_set_sensor_trip_point {
52 __le32 id;
53 __le32 event_control;
54#define SENSOR_TP_EVENT_MASK (0x3)
55#define SENSOR_TP_DISABLED 0x0
56#define SENSOR_TP_POSITIVE 0x1
57#define SENSOR_TP_NEGATIVE 0x2
58#define SENSOR_TP_BOTH 0x3
59#define SENSOR_TP_ID(x) (((x) & 0xff) << 4)
60 __le32 value_low;
61 __le32 value_high;
62};
63
64struct scmi_msg_sensor_reading_get {
65 __le32 id;
66 __le32 flags;
67#define SENSOR_READ_ASYNC BIT(0)
68};
69
70struct sensors_info {
71 int num_sensors;
72 int max_requests;
73 u64 reg_addr;
74 u32 reg_size;
75 struct scmi_sensor_info *sensors;
76};
77
78static int scmi_sensor_attributes_get(const struct scmi_handle *handle,
79 struct sensors_info *si)
80{
81 int ret;
82 struct scmi_xfer *t;
83 struct scmi_msg_resp_sensor_attributes *attr;
84
85 ret = scmi_xfer_get_init(handle, PROTOCOL_ATTRIBUTES,
86 SCMI_PROTOCOL_SENSOR, 0, sizeof(*attr), &t);
87 if (ret)
88 return ret;
89
90 attr = t->rx.buf;
91
92 ret = scmi_do_xfer(handle, t);
93 if (!ret) {
94 si->num_sensors = le16_to_cpu(attr->num_sensors);
95 si->max_requests = attr->max_requests;
96 si->reg_addr = le32_to_cpu(attr->reg_addr_low) |
97 (u64)le32_to_cpu(attr->reg_addr_high) << 32;
98 si->reg_size = le32_to_cpu(attr->reg_size);
99 }
100
101 scmi_xfer_put(handle, t);
102 return ret;
103}
104
105static int scmi_sensor_description_get(const struct scmi_handle *handle,
106 struct sensors_info *si)
107{
108 int ret, cnt;
109 u32 desc_index = 0;
110 u16 num_returned, num_remaining;
111 struct scmi_xfer *t;
112 struct scmi_msg_resp_sensor_description *buf;
113
114 ret = scmi_xfer_get_init(handle, SENSOR_DESCRIPTION_GET,
115 SCMI_PROTOCOL_SENSOR, sizeof(__le32), 0, &t);
116 if (ret)
117 return ret;
118
119 buf = t->rx.buf;
120
121 do {
122 /* Set the number of sensors to be skipped/already read */
123 put_unaligned_le32(desc_index, t->tx.buf);
124
125 ret = scmi_do_xfer(handle, t);
126 if (ret)
127 break;
128
129 num_returned = le16_to_cpu(buf->num_returned);
130 num_remaining = le16_to_cpu(buf->num_remaining);
131
132 if (desc_index + num_returned > si->num_sensors) {
133 dev_err(handle->dev, "No. of sensors can't exceed %d",
134 si->num_sensors);
135 break;
136 }
137
138 for (cnt = 0; cnt < num_returned; cnt++) {
139 u32 attrh, attrl;
140 struct scmi_sensor_info *s;
141
142 attrl = le32_to_cpu(buf->desc[cnt].attributes_low);
143 attrh = le32_to_cpu(buf->desc[cnt].attributes_high);
144 s = &si->sensors[desc_index + cnt];
145 s->id = le32_to_cpu(buf->desc[cnt].id);
146 s->type = SENSOR_TYPE(attrh);
147 s->scale = SENSOR_SCALE(attrh);
148 /* Sign extend to a full s8 */
149 if (s->scale & SENSOR_SCALE_SIGN)
150 s->scale |= SENSOR_SCALE_EXTEND;
151 s->async = SUPPORTS_ASYNC_READ(attrl);
152 s->num_trip_points = NUM_TRIP_POINTS(attrl);
153 strlcpy(s->name, buf->desc[cnt].name, SCMI_MAX_STR_SIZE);
154 }
155
156 desc_index += num_returned;
157 /*
158 * check for both returned and remaining to avoid infinite
159 * loop due to buggy firmware
160 */
161 } while (num_returned && num_remaining);
162
163 scmi_xfer_put(handle, t);
164 return ret;
165}
166
167static int scmi_sensor_trip_point_notify(const struct scmi_handle *handle,
168 u32 sensor_id, bool enable)
169{
170 int ret;
171 u32 evt_cntl = enable ? SENSOR_TP_NOTIFY_ALL : 0;
172 struct scmi_xfer *t;
173 struct scmi_msg_sensor_trip_point_notify *cfg;
174
175 ret = scmi_xfer_get_init(handle, SENSOR_TRIP_POINT_NOTIFY,
176 SCMI_PROTOCOL_SENSOR, sizeof(*cfg), 0, &t);
177 if (ret)
178 return ret;
179
180 cfg = t->tx.buf;
181 cfg->id = cpu_to_le32(sensor_id);
182 cfg->event_control = cpu_to_le32(evt_cntl);
183
184 ret = scmi_do_xfer(handle, t);
185
186 scmi_xfer_put(handle, t);
187 return ret;
188}
189
190static int
191scmi_sensor_trip_point_config(const struct scmi_handle *handle, u32 sensor_id,
192 u8 trip_id, u64 trip_value)
193{
194 int ret;
195 u32 evt_cntl = SENSOR_TP_BOTH;
196 struct scmi_xfer *t;
197 struct scmi_msg_set_sensor_trip_point *trip;
198
199 ret = scmi_xfer_get_init(handle, SENSOR_TRIP_POINT_CONFIG,
200 SCMI_PROTOCOL_SENSOR, sizeof(*trip), 0, &t);
201 if (ret)
202 return ret;
203
204 trip = t->tx.buf;
205 trip->id = cpu_to_le32(sensor_id);
206 trip->event_control = cpu_to_le32(evt_cntl | SENSOR_TP_ID(trip_id));
207 trip->value_low = cpu_to_le32(trip_value & 0xffffffff);
208 trip->value_high = cpu_to_le32(trip_value >> 32);
209
210 ret = scmi_do_xfer(handle, t);
211
212 scmi_xfer_put(handle, t);
213 return ret;
214}
215
216static int scmi_sensor_reading_get(const struct scmi_handle *handle,
217 u32 sensor_id, u64 *value)
218{
219 int ret;
220 struct scmi_xfer *t;
221 struct scmi_msg_sensor_reading_get *sensor;
222 struct sensors_info *si = handle->sensor_priv;
223 struct scmi_sensor_info *s = si->sensors + sensor_id;
224
225 ret = scmi_xfer_get_init(handle, SENSOR_READING_GET,
226 SCMI_PROTOCOL_SENSOR, sizeof(*sensor),
227 sizeof(u64), &t);
228 if (ret)
229 return ret;
230
231 sensor = t->tx.buf;
232 sensor->id = cpu_to_le32(sensor_id);
233
234 if (s->async) {
235 sensor->flags = cpu_to_le32(SENSOR_READ_ASYNC);
236 ret = scmi_do_xfer_with_response(handle, t);
237 if (!ret)
238 *value = get_unaligned_le64((void *)
239 ((__le32 *)t->rx.buf + 1));
240 } else {
241 sensor->flags = cpu_to_le32(0);
242 ret = scmi_do_xfer(handle, t);
243 if (!ret)
244 *value = get_unaligned_le64(t->rx.buf);
245 }
246
247 scmi_xfer_put(handle, t);
248 return ret;
249}
250
251static const struct scmi_sensor_info *
252scmi_sensor_info_get(const struct scmi_handle *handle, u32 sensor_id)
253{
254 struct sensors_info *si = handle->sensor_priv;
255
256 return si->sensors + sensor_id;
257}
258
259static int scmi_sensor_count_get(const struct scmi_handle *handle)
260{
261 struct sensors_info *si = handle->sensor_priv;
262
263 return si->num_sensors;
264}
265
266static struct scmi_sensor_ops sensor_ops = {
267 .count_get = scmi_sensor_count_get,
268 .info_get = scmi_sensor_info_get,
269 .trip_point_notify = scmi_sensor_trip_point_notify,
270 .trip_point_config = scmi_sensor_trip_point_config,
271 .reading_get = scmi_sensor_reading_get,
272};
273
274static int scmi_sensors_protocol_init(struct scmi_handle *handle)
275{
276 u32 version;
277 struct sensors_info *sinfo;
278
279 scmi_version_get(handle, SCMI_PROTOCOL_SENSOR, &version);
280
281 dev_dbg(handle->dev, "Sensor Version %d.%d\n",
282 PROTOCOL_REV_MAJOR(version), PROTOCOL_REV_MINOR(version));
283
284 sinfo = devm_kzalloc(handle->dev, sizeof(*sinfo), GFP_KERNEL);
285 if (!sinfo)
286 return -ENOMEM;
287
288 scmi_sensor_attributes_get(handle, sinfo);
289
290 sinfo->sensors = devm_kcalloc(handle->dev, sinfo->num_sensors,
291 sizeof(*sinfo->sensors), GFP_KERNEL);
292 if (!sinfo->sensors)
293 return -ENOMEM;
294
295 scmi_sensor_description_get(handle, sinfo);
296
297 handle->sensor_ops = &sensor_ops;
298 handle->sensor_priv = sinfo;
299
300 return 0;
301}
302
303static int __init scmi_sensors_init(void)
304{
305 return scmi_protocol_register(SCMI_PROTOCOL_SENSOR,
306 &scmi_sensors_protocol_init);
307}
308subsys_initcall(scmi_sensors_init);