xref: /linux/drivers/iio/orientation/hid-sensor-incl-3d.c (revision 0ea5c948cb64bab5bc7a5516774eb8536f05aa0d)
1f6cc69f1SThomas Gleixner // SPDX-License-Identifier: GPL-2.0-only
2098d3becSSrinivas Pandruvada /*
3098d3becSSrinivas Pandruvada  * HID Sensors Driver
4098d3becSSrinivas Pandruvada  * Copyright (c) 2013, Intel Corporation.
5098d3becSSrinivas Pandruvada  */
6098d3becSSrinivas Pandruvada 
7098d3becSSrinivas Pandruvada #include <linux/device.h>
8098d3becSSrinivas Pandruvada #include <linux/platform_device.h>
9098d3becSSrinivas Pandruvada #include <linux/module.h>
10fb226ae7SJonathan Cameron #include <linux/mod_devicetable.h>
11098d3becSSrinivas Pandruvada #include <linux/slab.h>
12098d3becSSrinivas Pandruvada #include <linux/hid-sensor-hub.h>
13098d3becSSrinivas Pandruvada #include <linux/iio/iio.h>
14098d3becSSrinivas Pandruvada #include <linux/iio/buffer.h>
15098d3becSSrinivas Pandruvada #include "../common/hid-sensors/hid-sensor-trigger.h"
16098d3becSSrinivas Pandruvada 
17098d3becSSrinivas Pandruvada enum incl_3d_channel {
18098d3becSSrinivas Pandruvada 	CHANNEL_SCAN_INDEX_X,
19098d3becSSrinivas Pandruvada 	CHANNEL_SCAN_INDEX_Y,
20098d3becSSrinivas Pandruvada 	CHANNEL_SCAN_INDEX_Z,
21098d3becSSrinivas Pandruvada 	INCLI_3D_CHANNEL_MAX,
22098d3becSSrinivas Pandruvada };
23098d3becSSrinivas Pandruvada 
2404fe70d1SYe Xiang #define CHANNEL_SCAN_INDEX_TIMESTAMP INCLI_3D_CHANNEL_MAX
2504fe70d1SYe Xiang 
26098d3becSSrinivas Pandruvada struct incl_3d_state {
27098d3becSSrinivas Pandruvada 	struct hid_sensor_hub_callbacks callbacks;
28098d3becSSrinivas Pandruvada 	struct hid_sensor_common common_attributes;
29098d3becSSrinivas Pandruvada 	struct hid_sensor_hub_attribute_info incl[INCLI_3D_CHANNEL_MAX];
3004fe70d1SYe Xiang 	struct {
31098d3becSSrinivas Pandruvada 		u32 incl_val[INCLI_3D_CHANNEL_MAX];
3204fe70d1SYe Xiang 		u64 timestamp __aligned(8);
3304fe70d1SYe Xiang 	} scan;
34be8e48d6SSrinivas Pandruvada 	int scale_pre_decml;
35be8e48d6SSrinivas Pandruvada 	int scale_post_decml;
36be8e48d6SSrinivas Pandruvada 	int scale_precision;
37be8e48d6SSrinivas Pandruvada 	int value_offset;
3804fe70d1SYe Xiang 	s64 timestamp;
39098d3becSSrinivas Pandruvada };
40098d3becSSrinivas Pandruvada 
41098d3becSSrinivas Pandruvada static const u32 incl_3d_addresses[INCLI_3D_CHANNEL_MAX] = {
42098d3becSSrinivas Pandruvada 	HID_USAGE_SENSOR_ORIENT_TILT_X,
43098d3becSSrinivas Pandruvada 	HID_USAGE_SENSOR_ORIENT_TILT_Y,
44098d3becSSrinivas Pandruvada 	HID_USAGE_SENSOR_ORIENT_TILT_Z
45098d3becSSrinivas Pandruvada };
46098d3becSSrinivas Pandruvada 
470e41fd51SYe Xiang static const u32 incl_3d_sensitivity_addresses[] = {
480e41fd51SYe Xiang 	HID_USAGE_SENSOR_DATA_ORIENTATION,
494efd13c3SYe Xiang 	HID_USAGE_SENSOR_ORIENT_TILT,
500e41fd51SYe Xiang };
510e41fd51SYe Xiang 
52098d3becSSrinivas Pandruvada /* Channel definitions */
53098d3becSSrinivas Pandruvada static const struct iio_chan_spec incl_3d_channels[] = {
54098d3becSSrinivas Pandruvada 	{
55098d3becSSrinivas Pandruvada 		.type = IIO_INCLI,
56098d3becSSrinivas Pandruvada 		.modified = 1,
57098d3becSSrinivas Pandruvada 		.channel2 = IIO_MOD_X,
58098d3becSSrinivas Pandruvada 		.info_mask_separate = BIT(IIO_CHAN_INFO_RAW),
59098d3becSSrinivas Pandruvada 		.info_mask_shared_by_type = BIT(IIO_CHAN_INFO_OFFSET) |
60098d3becSSrinivas Pandruvada 		BIT(IIO_CHAN_INFO_SCALE) |
61098d3becSSrinivas Pandruvada 		BIT(IIO_CHAN_INFO_SAMP_FREQ) |
62098d3becSSrinivas Pandruvada 		BIT(IIO_CHAN_INFO_HYSTERESIS),
63098d3becSSrinivas Pandruvada 		.scan_index = CHANNEL_SCAN_INDEX_X,
64098d3becSSrinivas Pandruvada 	}, {
65098d3becSSrinivas Pandruvada 		.type = IIO_INCLI,
66098d3becSSrinivas Pandruvada 		.modified = 1,
67098d3becSSrinivas Pandruvada 		.channel2 = IIO_MOD_Y,
68098d3becSSrinivas Pandruvada 		.info_mask_separate = BIT(IIO_CHAN_INFO_RAW),
69098d3becSSrinivas Pandruvada 		.info_mask_shared_by_type = BIT(IIO_CHAN_INFO_OFFSET) |
70098d3becSSrinivas Pandruvada 		BIT(IIO_CHAN_INFO_SCALE) |
71098d3becSSrinivas Pandruvada 		BIT(IIO_CHAN_INFO_SAMP_FREQ) |
72098d3becSSrinivas Pandruvada 		BIT(IIO_CHAN_INFO_HYSTERESIS),
73098d3becSSrinivas Pandruvada 		.scan_index = CHANNEL_SCAN_INDEX_Y,
74098d3becSSrinivas Pandruvada 	}, {
75098d3becSSrinivas Pandruvada 		.type = IIO_INCLI,
76098d3becSSrinivas Pandruvada 		.modified = 1,
77098d3becSSrinivas Pandruvada 		.channel2 = IIO_MOD_Z,
78098d3becSSrinivas Pandruvada 		.info_mask_separate = BIT(IIO_CHAN_INFO_RAW),
79098d3becSSrinivas Pandruvada 		.info_mask_shared_by_type = BIT(IIO_CHAN_INFO_OFFSET) |
80098d3becSSrinivas Pandruvada 		BIT(IIO_CHAN_INFO_SCALE) |
81098d3becSSrinivas Pandruvada 		BIT(IIO_CHAN_INFO_SAMP_FREQ) |
82098d3becSSrinivas Pandruvada 		BIT(IIO_CHAN_INFO_HYSTERESIS),
83098d3becSSrinivas Pandruvada 		.scan_index = CHANNEL_SCAN_INDEX_Z,
8404fe70d1SYe Xiang 	},
8504fe70d1SYe Xiang 	IIO_CHAN_SOFT_TIMESTAMP(CHANNEL_SCAN_INDEX_TIMESTAMP),
86098d3becSSrinivas Pandruvada };
87098d3becSSrinivas Pandruvada 
88098d3becSSrinivas Pandruvada /* Adjust channel real bits based on report descriptor */
incl_3d_adjust_channel_bit_mask(struct iio_chan_spec * chan,int size)89098d3becSSrinivas Pandruvada static void incl_3d_adjust_channel_bit_mask(struct iio_chan_spec *chan,
90098d3becSSrinivas Pandruvada 						int size)
91098d3becSSrinivas Pandruvada {
92098d3becSSrinivas Pandruvada 	chan->scan_type.sign = 's';
93098d3becSSrinivas Pandruvada 	/* Real storage bits will change based on the report desc. */
94098d3becSSrinivas Pandruvada 	chan->scan_type.realbits = size * 8;
95098d3becSSrinivas Pandruvada 	/* Maximum size of a sample to capture is u32 */
96098d3becSSrinivas Pandruvada 	chan->scan_type.storagebits = sizeof(u32) * 8;
97098d3becSSrinivas Pandruvada }
98098d3becSSrinivas Pandruvada 
99098d3becSSrinivas Pandruvada /* Channel read_raw handler */
incl_3d_read_raw(struct iio_dev * indio_dev,struct iio_chan_spec const * chan,int * val,int * val2,long mask)100098d3becSSrinivas Pandruvada static int incl_3d_read_raw(struct iio_dev *indio_dev,
101098d3becSSrinivas Pandruvada 			      struct iio_chan_spec const *chan,
102098d3becSSrinivas Pandruvada 			      int *val, int *val2,
103098d3becSSrinivas Pandruvada 			      long mask)
104098d3becSSrinivas Pandruvada {
105098d3becSSrinivas Pandruvada 	struct incl_3d_state *incl_state = iio_priv(indio_dev);
106098d3becSSrinivas Pandruvada 	int report_id = -1;
107098d3becSSrinivas Pandruvada 	u32 address;
108098d3becSSrinivas Pandruvada 	int ret_type;
1090145b505SHans de Goede 	s32 min;
110098d3becSSrinivas Pandruvada 
111098d3becSSrinivas Pandruvada 	*val = 0;
112098d3becSSrinivas Pandruvada 	*val2 = 0;
113098d3becSSrinivas Pandruvada 	switch (mask) {
114098d3becSSrinivas Pandruvada 	case IIO_CHAN_INFO_RAW:
1156fe588e5SSrinivas Pandruvada 		hid_sensor_power_state(&incl_state->common_attributes, true);
1160145b505SHans de Goede 		report_id = incl_state->incl[chan->scan_index].report_id;
1170145b505SHans de Goede 		min = incl_state->incl[chan->scan_index].logical_minimum;
118098d3becSSrinivas Pandruvada 		address = incl_3d_addresses[chan->scan_index];
119098d3becSSrinivas Pandruvada 		if (report_id >= 0)
120098d3becSSrinivas Pandruvada 			*val = sensor_hub_input_attr_get_raw_value(
121098d3becSSrinivas Pandruvada 				incl_state->common_attributes.hsdev,
122098d3becSSrinivas Pandruvada 				HID_USAGE_SENSOR_INCLINOMETER_3D, address,
123b3f4737dSSrinivas Pandruvada 				report_id,
1240145b505SHans de Goede 				SENSOR_HUB_SYNC,
1250145b505SHans de Goede 				min < 0);
126098d3becSSrinivas Pandruvada 		else {
1276fe588e5SSrinivas Pandruvada 			hid_sensor_power_state(&incl_state->common_attributes,
1286fe588e5SSrinivas Pandruvada 						false);
129098d3becSSrinivas Pandruvada 			return -EINVAL;
130098d3becSSrinivas Pandruvada 		}
1316fe588e5SSrinivas Pandruvada 		hid_sensor_power_state(&incl_state->common_attributes, false);
132098d3becSSrinivas Pandruvada 		ret_type = IIO_VAL_INT;
133098d3becSSrinivas Pandruvada 		break;
134098d3becSSrinivas Pandruvada 	case IIO_CHAN_INFO_SCALE:
135be8e48d6SSrinivas Pandruvada 		*val = incl_state->scale_pre_decml;
136be8e48d6SSrinivas Pandruvada 		*val2 = incl_state->scale_post_decml;
137be8e48d6SSrinivas Pandruvada 		ret_type = incl_state->scale_precision;
138098d3becSSrinivas Pandruvada 		break;
139098d3becSSrinivas Pandruvada 	case IIO_CHAN_INFO_OFFSET:
140be8e48d6SSrinivas Pandruvada 		*val = incl_state->value_offset;
141098d3becSSrinivas Pandruvada 		ret_type = IIO_VAL_INT;
142098d3becSSrinivas Pandruvada 		break;
143098d3becSSrinivas Pandruvada 	case IIO_CHAN_INFO_SAMP_FREQ:
144098d3becSSrinivas Pandruvada 		ret_type = hid_sensor_read_samp_freq_value(
145098d3becSSrinivas Pandruvada 			&incl_state->common_attributes, val, val2);
146098d3becSSrinivas Pandruvada 		break;
147098d3becSSrinivas Pandruvada 	case IIO_CHAN_INFO_HYSTERESIS:
148098d3becSSrinivas Pandruvada 		ret_type = hid_sensor_read_raw_hyst_value(
149098d3becSSrinivas Pandruvada 			&incl_state->common_attributes, val, val2);
150098d3becSSrinivas Pandruvada 		break;
151098d3becSSrinivas Pandruvada 	default:
152098d3becSSrinivas Pandruvada 		ret_type = -EINVAL;
153098d3becSSrinivas Pandruvada 		break;
154098d3becSSrinivas Pandruvada 	}
155098d3becSSrinivas Pandruvada 
156098d3becSSrinivas Pandruvada 	return ret_type;
157098d3becSSrinivas Pandruvada }
158098d3becSSrinivas Pandruvada 
159098d3becSSrinivas Pandruvada /* Channel write_raw handler */
incl_3d_write_raw(struct iio_dev * indio_dev,struct iio_chan_spec const * chan,int val,int val2,long mask)160098d3becSSrinivas Pandruvada static int incl_3d_write_raw(struct iio_dev *indio_dev,
161098d3becSSrinivas Pandruvada 			       struct iio_chan_spec const *chan,
162098d3becSSrinivas Pandruvada 			       int val,
163098d3becSSrinivas Pandruvada 			       int val2,
164098d3becSSrinivas Pandruvada 			       long mask)
165098d3becSSrinivas Pandruvada {
166098d3becSSrinivas Pandruvada 	struct incl_3d_state *incl_state = iio_priv(indio_dev);
167098d3becSSrinivas Pandruvada 	int ret;
168098d3becSSrinivas Pandruvada 
169098d3becSSrinivas Pandruvada 	switch (mask) {
170098d3becSSrinivas Pandruvada 	case IIO_CHAN_INFO_SAMP_FREQ:
171098d3becSSrinivas Pandruvada 		ret = hid_sensor_write_samp_freq_value(
172098d3becSSrinivas Pandruvada 				&incl_state->common_attributes, val, val2);
173098d3becSSrinivas Pandruvada 		break;
174098d3becSSrinivas Pandruvada 	case IIO_CHAN_INFO_HYSTERESIS:
175098d3becSSrinivas Pandruvada 		ret = hid_sensor_write_raw_hyst_value(
176098d3becSSrinivas Pandruvada 				&incl_state->common_attributes, val, val2);
177098d3becSSrinivas Pandruvada 		break;
178098d3becSSrinivas Pandruvada 	default:
179098d3becSSrinivas Pandruvada 		ret = -EINVAL;
180098d3becSSrinivas Pandruvada 	}
181098d3becSSrinivas Pandruvada 
182098d3becSSrinivas Pandruvada 	return ret;
183098d3becSSrinivas Pandruvada }
184098d3becSSrinivas Pandruvada 
185098d3becSSrinivas Pandruvada static const struct iio_info incl_3d_info = {
186098d3becSSrinivas Pandruvada 	.read_raw = &incl_3d_read_raw,
187098d3becSSrinivas Pandruvada 	.write_raw = &incl_3d_write_raw,
188098d3becSSrinivas Pandruvada };
189098d3becSSrinivas Pandruvada 
190098d3becSSrinivas Pandruvada /* Callback handler to send event after all samples are received and captured */
incl_3d_proc_event(struct hid_sensor_hub_device * hsdev,unsigned usage_id,void * priv)191098d3becSSrinivas Pandruvada static int incl_3d_proc_event(struct hid_sensor_hub_device *hsdev,
192098d3becSSrinivas Pandruvada 				unsigned usage_id,
193098d3becSSrinivas Pandruvada 				void *priv)
194098d3becSSrinivas Pandruvada {
195098d3becSSrinivas Pandruvada 	struct iio_dev *indio_dev = platform_get_drvdata(priv);
196098d3becSSrinivas Pandruvada 	struct incl_3d_state *incl_state = iio_priv(indio_dev);
197098d3becSSrinivas Pandruvada 
19856ff6be6SSrinivas Pandruvada 	dev_dbg(&indio_dev->dev, "incl_3d_proc_event\n");
19904fe70d1SYe Xiang 	if (atomic_read(&incl_state->common_attributes.data_ready)) {
20004fe70d1SYe Xiang 		if (!incl_state->timestamp)
20104fe70d1SYe Xiang 			incl_state->timestamp = iio_get_time_ns(indio_dev);
20204fe70d1SYe Xiang 
20304fe70d1SYe Xiang 		iio_push_to_buffers_with_timestamp(indio_dev,
20404fe70d1SYe Xiang 						   &incl_state->scan,
20504fe70d1SYe Xiang 						   incl_state->timestamp);
20604fe70d1SYe Xiang 
20704fe70d1SYe Xiang 		incl_state->timestamp = 0;
20804fe70d1SYe Xiang 	}
209098d3becSSrinivas Pandruvada 
210098d3becSSrinivas Pandruvada 	return 0;
211098d3becSSrinivas Pandruvada }
212098d3becSSrinivas Pandruvada 
213098d3becSSrinivas Pandruvada /* Capture samples in local storage */
incl_3d_capture_sample(struct hid_sensor_hub_device * hsdev,unsigned usage_id,size_t raw_len,char * raw_data,void * priv)214098d3becSSrinivas Pandruvada static int incl_3d_capture_sample(struct hid_sensor_hub_device *hsdev,
215098d3becSSrinivas Pandruvada 				unsigned usage_id,
216098d3becSSrinivas Pandruvada 				size_t raw_len, char *raw_data,
217098d3becSSrinivas Pandruvada 				void *priv)
218098d3becSSrinivas Pandruvada {
219098d3becSSrinivas Pandruvada 	struct iio_dev *indio_dev = platform_get_drvdata(priv);
220098d3becSSrinivas Pandruvada 	struct incl_3d_state *incl_state = iio_priv(indio_dev);
221098d3becSSrinivas Pandruvada 	int ret = 0;
222098d3becSSrinivas Pandruvada 
223098d3becSSrinivas Pandruvada 	switch (usage_id) {
224098d3becSSrinivas Pandruvada 	case HID_USAGE_SENSOR_ORIENT_TILT_X:
22504fe70d1SYe Xiang 		incl_state->scan.incl_val[CHANNEL_SCAN_INDEX_X] = *(u32 *)raw_data;
226098d3becSSrinivas Pandruvada 	break;
227098d3becSSrinivas Pandruvada 	case HID_USAGE_SENSOR_ORIENT_TILT_Y:
22804fe70d1SYe Xiang 		incl_state->scan.incl_val[CHANNEL_SCAN_INDEX_Y] = *(u32 *)raw_data;
229098d3becSSrinivas Pandruvada 	break;
230098d3becSSrinivas Pandruvada 	case HID_USAGE_SENSOR_ORIENT_TILT_Z:
23104fe70d1SYe Xiang 		incl_state->scan.incl_val[CHANNEL_SCAN_INDEX_Z] = *(u32 *)raw_data;
23204fe70d1SYe Xiang 	break;
23304fe70d1SYe Xiang 	case HID_USAGE_SENSOR_TIME_TIMESTAMP:
23404fe70d1SYe Xiang 		incl_state->timestamp =
23504fe70d1SYe Xiang 			hid_sensor_convert_timestamp(&incl_state->common_attributes,
23604fe70d1SYe Xiang 						     *(s64 *)raw_data);
237098d3becSSrinivas Pandruvada 	break;
238098d3becSSrinivas Pandruvada 	default:
239098d3becSSrinivas Pandruvada 		ret = -EINVAL;
240098d3becSSrinivas Pandruvada 		break;
241098d3becSSrinivas Pandruvada 	}
242098d3becSSrinivas Pandruvada 
243098d3becSSrinivas Pandruvada 	return ret;
244098d3becSSrinivas Pandruvada }
245098d3becSSrinivas Pandruvada 
246098d3becSSrinivas Pandruvada /* Parse report which is specific to an usage id*/
incl_3d_parse_report(struct platform_device * pdev,struct hid_sensor_hub_device * hsdev,struct iio_chan_spec * channels,unsigned usage_id,struct incl_3d_state * st)247098d3becSSrinivas Pandruvada static int incl_3d_parse_report(struct platform_device *pdev,
248098d3becSSrinivas Pandruvada 				struct hid_sensor_hub_device *hsdev,
249098d3becSSrinivas Pandruvada 				struct iio_chan_spec *channels,
250098d3becSSrinivas Pandruvada 				unsigned usage_id,
251098d3becSSrinivas Pandruvada 				struct incl_3d_state *st)
252098d3becSSrinivas Pandruvada {
253098d3becSSrinivas Pandruvada 	int ret;
254098d3becSSrinivas Pandruvada 
255098d3becSSrinivas Pandruvada 	ret = sensor_hub_input_get_attribute_info(hsdev,
256098d3becSSrinivas Pandruvada 				HID_INPUT_REPORT,
257098d3becSSrinivas Pandruvada 				usage_id,
258098d3becSSrinivas Pandruvada 				HID_USAGE_SENSOR_ORIENT_TILT_X,
259098d3becSSrinivas Pandruvada 				&st->incl[CHANNEL_SCAN_INDEX_X]);
260098d3becSSrinivas Pandruvada 	if (ret)
261098d3becSSrinivas Pandruvada 		return ret;
262098d3becSSrinivas Pandruvada 	incl_3d_adjust_channel_bit_mask(&channels[CHANNEL_SCAN_INDEX_X],
263098d3becSSrinivas Pandruvada 				st->incl[CHANNEL_SCAN_INDEX_X].size);
264098d3becSSrinivas Pandruvada 
265098d3becSSrinivas Pandruvada 	ret = sensor_hub_input_get_attribute_info(hsdev,
266098d3becSSrinivas Pandruvada 				HID_INPUT_REPORT,
267098d3becSSrinivas Pandruvada 				usage_id,
268098d3becSSrinivas Pandruvada 				HID_USAGE_SENSOR_ORIENT_TILT_Y,
269098d3becSSrinivas Pandruvada 				&st->incl[CHANNEL_SCAN_INDEX_Y]);
270098d3becSSrinivas Pandruvada 	if (ret)
271098d3becSSrinivas Pandruvada 		return ret;
272098d3becSSrinivas Pandruvada 	incl_3d_adjust_channel_bit_mask(&channels[CHANNEL_SCAN_INDEX_Y],
273098d3becSSrinivas Pandruvada 				st->incl[CHANNEL_SCAN_INDEX_Y].size);
274098d3becSSrinivas Pandruvada 
275098d3becSSrinivas Pandruvada 	ret = sensor_hub_input_get_attribute_info(hsdev,
276098d3becSSrinivas Pandruvada 				HID_INPUT_REPORT,
277098d3becSSrinivas Pandruvada 				usage_id,
278098d3becSSrinivas Pandruvada 				HID_USAGE_SENSOR_ORIENT_TILT_Z,
279098d3becSSrinivas Pandruvada 				&st->incl[CHANNEL_SCAN_INDEX_Z]);
280098d3becSSrinivas Pandruvada 	if (ret)
281098d3becSSrinivas Pandruvada 		return ret;
282098d3becSSrinivas Pandruvada 	incl_3d_adjust_channel_bit_mask(&channels[CHANNEL_SCAN_INDEX_Z],
283098d3becSSrinivas Pandruvada 				st->incl[CHANNEL_SCAN_INDEX_Z].size);
284098d3becSSrinivas Pandruvada 
285098d3becSSrinivas Pandruvada 	dev_dbg(&pdev->dev, "incl_3d %x:%x, %x:%x, %x:%x\n",
286098d3becSSrinivas Pandruvada 			st->incl[0].index,
287098d3becSSrinivas Pandruvada 			st->incl[0].report_id,
288098d3becSSrinivas Pandruvada 			st->incl[1].index, st->incl[1].report_id,
289098d3becSSrinivas Pandruvada 			st->incl[2].index, st->incl[2].report_id);
290098d3becSSrinivas Pandruvada 
291be8e48d6SSrinivas Pandruvada 	st->scale_precision = hid_sensor_format_scale(
292be8e48d6SSrinivas Pandruvada 				HID_USAGE_SENSOR_INCLINOMETER_3D,
293be8e48d6SSrinivas Pandruvada 				&st->incl[CHANNEL_SCAN_INDEX_X],
294be8e48d6SSrinivas Pandruvada 				&st->scale_pre_decml, &st->scale_post_decml);
295be8e48d6SSrinivas Pandruvada 
296098d3becSSrinivas Pandruvada 	return ret;
297098d3becSSrinivas Pandruvada }
298098d3becSSrinivas Pandruvada 
299098d3becSSrinivas Pandruvada /* Function to initialize the processing for usage id */
hid_incl_3d_probe(struct platform_device * pdev)300098d3becSSrinivas Pandruvada static int hid_incl_3d_probe(struct platform_device *pdev)
301098d3becSSrinivas Pandruvada {
302098d3becSSrinivas Pandruvada 	int ret;
303098d3becSSrinivas Pandruvada 	static char *name = "incli_3d";
304098d3becSSrinivas Pandruvada 	struct iio_dev *indio_dev;
305098d3becSSrinivas Pandruvada 	struct incl_3d_state *incl_state;
306098d3becSSrinivas Pandruvada 	struct hid_sensor_hub_device *hsdev = pdev->dev.platform_data;
307098d3becSSrinivas Pandruvada 
308098d3becSSrinivas Pandruvada 	indio_dev = devm_iio_device_alloc(&pdev->dev,
309098d3becSSrinivas Pandruvada 					  sizeof(struct incl_3d_state));
310098d3becSSrinivas Pandruvada 	if (indio_dev == NULL)
311098d3becSSrinivas Pandruvada 		return -ENOMEM;
312098d3becSSrinivas Pandruvada 
313098d3becSSrinivas Pandruvada 	platform_set_drvdata(pdev, indio_dev);
314098d3becSSrinivas Pandruvada 
315098d3becSSrinivas Pandruvada 	incl_state = iio_priv(indio_dev);
316098d3becSSrinivas Pandruvada 	incl_state->common_attributes.hsdev = hsdev;
317098d3becSSrinivas Pandruvada 	incl_state->common_attributes.pdev = pdev;
318098d3becSSrinivas Pandruvada 
319098d3becSSrinivas Pandruvada 	ret = hid_sensor_parse_common_attributes(hsdev,
320098d3becSSrinivas Pandruvada 				HID_USAGE_SENSOR_INCLINOMETER_3D,
3210e41fd51SYe Xiang 				&incl_state->common_attributes,
3220e41fd51SYe Xiang 				incl_3d_sensitivity_addresses,
3230e41fd51SYe Xiang 				ARRAY_SIZE(incl_3d_sensitivity_addresses));
324098d3becSSrinivas Pandruvada 	if (ret) {
325098d3becSSrinivas Pandruvada 		dev_err(&pdev->dev, "failed to setup common attributes\n");
326098d3becSSrinivas Pandruvada 		return ret;
327098d3becSSrinivas Pandruvada 	}
328098d3becSSrinivas Pandruvada 
3298b2ac516SAlexandru Ardelean 	indio_dev->channels = devm_kmemdup(&pdev->dev, incl_3d_channels,
33032ee56e3SFabio Estevam 					   sizeof(incl_3d_channels), GFP_KERNEL);
33132ee56e3SFabio Estevam 	if (!indio_dev->channels) {
332098d3becSSrinivas Pandruvada 		dev_err(&pdev->dev, "failed to duplicate channels\n");
333098d3becSSrinivas Pandruvada 		return -ENOMEM;
334098d3becSSrinivas Pandruvada 	}
335098d3becSSrinivas Pandruvada 
33632ee56e3SFabio Estevam 	ret = incl_3d_parse_report(pdev, hsdev,
33732ee56e3SFabio Estevam 				   (struct iio_chan_spec *)indio_dev->channels,
33832ee56e3SFabio Estevam 				   HID_USAGE_SENSOR_INCLINOMETER_3D,
33932ee56e3SFabio Estevam 				   incl_state);
340098d3becSSrinivas Pandruvada 	if (ret) {
341098d3becSSrinivas Pandruvada 		dev_err(&pdev->dev, "failed to setup attributes\n");
3428b2ac516SAlexandru Ardelean 		return ret;
343098d3becSSrinivas Pandruvada 	}
344098d3becSSrinivas Pandruvada 
345098d3becSSrinivas Pandruvada 	indio_dev->num_channels = ARRAY_SIZE(incl_3d_channels);
346098d3becSSrinivas Pandruvada 	indio_dev->info = &incl_3d_info;
347098d3becSSrinivas Pandruvada 	indio_dev->name = name;
348098d3becSSrinivas Pandruvada 	indio_dev->modes = INDIO_DIRECT_MODE;
349098d3becSSrinivas Pandruvada 
35056ff6be6SSrinivas Pandruvada 	atomic_set(&incl_state->common_attributes.data_ready, 0);
351067fda1cSAlexandru Ardelean 
352098d3becSSrinivas Pandruvada 	ret = hid_sensor_setup_trigger(indio_dev, name,
353098d3becSSrinivas Pandruvada 					&incl_state->common_attributes);
354098d3becSSrinivas Pandruvada 	if (ret) {
355098d3becSSrinivas Pandruvada 		dev_err(&pdev->dev, "trigger setup failed\n");
3568b2ac516SAlexandru Ardelean 		return ret;
357098d3becSSrinivas Pandruvada 	}
358098d3becSSrinivas Pandruvada 
359098d3becSSrinivas Pandruvada 	ret = iio_device_register(indio_dev);
360098d3becSSrinivas Pandruvada 	if (ret) {
361098d3becSSrinivas Pandruvada 		dev_err(&pdev->dev, "device register failed\n");
362098d3becSSrinivas Pandruvada 		goto error_remove_trigger;
363098d3becSSrinivas Pandruvada 	}
364098d3becSSrinivas Pandruvada 
365098d3becSSrinivas Pandruvada 	incl_state->callbacks.send_event = incl_3d_proc_event;
366098d3becSSrinivas Pandruvada 	incl_state->callbacks.capture_sample = incl_3d_capture_sample;
367098d3becSSrinivas Pandruvada 	incl_state->callbacks.pdev = pdev;
368098d3becSSrinivas Pandruvada 	ret = sensor_hub_register_callback(hsdev,
369098d3becSSrinivas Pandruvada 					HID_USAGE_SENSOR_INCLINOMETER_3D,
370098d3becSSrinivas Pandruvada 					&incl_state->callbacks);
371098d3becSSrinivas Pandruvada 	if (ret) {
372098d3becSSrinivas Pandruvada 		dev_err(&pdev->dev, "callback reg failed\n");
373098d3becSSrinivas Pandruvada 		goto error_iio_unreg;
374098d3becSSrinivas Pandruvada 	}
375098d3becSSrinivas Pandruvada 
376098d3becSSrinivas Pandruvada 	return 0;
377098d3becSSrinivas Pandruvada 
378098d3becSSrinivas Pandruvada error_iio_unreg:
379098d3becSSrinivas Pandruvada 	iio_device_unregister(indio_dev);
380098d3becSSrinivas Pandruvada error_remove_trigger:
381067fda1cSAlexandru Ardelean 	hid_sensor_remove_trigger(indio_dev, &incl_state->common_attributes);
382098d3becSSrinivas Pandruvada 	return ret;
383098d3becSSrinivas Pandruvada }
384098d3becSSrinivas Pandruvada 
385098d3becSSrinivas Pandruvada /* Function to deinitialize the processing for usage id */
hid_incl_3d_remove(struct platform_device * pdev)386*44b25cf7SUwe Kleine-König static void hid_incl_3d_remove(struct platform_device *pdev)
387098d3becSSrinivas Pandruvada {
388098d3becSSrinivas Pandruvada 	struct hid_sensor_hub_device *hsdev = pdev->dev.platform_data;
389098d3becSSrinivas Pandruvada 	struct iio_dev *indio_dev = platform_get_drvdata(pdev);
390098d3becSSrinivas Pandruvada 	struct incl_3d_state *incl_state = iio_priv(indio_dev);
391098d3becSSrinivas Pandruvada 
392098d3becSSrinivas Pandruvada 	sensor_hub_remove_callback(hsdev, HID_USAGE_SENSOR_INCLINOMETER_3D);
393098d3becSSrinivas Pandruvada 	iio_device_unregister(indio_dev);
394067fda1cSAlexandru Ardelean 	hid_sensor_remove_trigger(indio_dev, &incl_state->common_attributes);
395098d3becSSrinivas Pandruvada }
396098d3becSSrinivas Pandruvada 
3975dd86df0SKrzysztof Kozlowski static const struct platform_device_id hid_incl_3d_ids[] = {
398098d3becSSrinivas Pandruvada 	{
399098d3becSSrinivas Pandruvada 		/* Format: HID-SENSOR-usage_id_in_hex_lowercase */
400098d3becSSrinivas Pandruvada 		.name = "HID-SENSOR-200086",
401098d3becSSrinivas Pandruvada 	},
402098d3becSSrinivas Pandruvada 	{ /* sentinel */ }
403098d3becSSrinivas Pandruvada };
404098d3becSSrinivas Pandruvada MODULE_DEVICE_TABLE(platform, hid_incl_3d_ids);
405098d3becSSrinivas Pandruvada 
406098d3becSSrinivas Pandruvada static struct platform_driver hid_incl_3d_platform_driver = {
407098d3becSSrinivas Pandruvada 	.id_table = hid_incl_3d_ids,
408098d3becSSrinivas Pandruvada 	.driver = {
409098d3becSSrinivas Pandruvada 		.name	= KBUILD_MODNAME,
410feb2b06dSSrinivas Pandruvada 		.pm	= &hid_sensor_pm_ops,
411098d3becSSrinivas Pandruvada 	},
412098d3becSSrinivas Pandruvada 	.probe		= hid_incl_3d_probe,
413*44b25cf7SUwe Kleine-König 	.remove_new	= hid_incl_3d_remove,
414098d3becSSrinivas Pandruvada };
415098d3becSSrinivas Pandruvada module_platform_driver(hid_incl_3d_platform_driver);
416098d3becSSrinivas Pandruvada 
417098d3becSSrinivas Pandruvada MODULE_DESCRIPTION("HID Sensor Inclinometer 3D");
418098d3becSSrinivas Pandruvada MODULE_AUTHOR("Srinivas Pandruvada <srinivas.pandruvada@linux.intel.com>");
419098d3becSSrinivas Pandruvada MODULE_LICENSE("GPL");
42012f13d1fSAndy Shevchenko MODULE_IMPORT_NS(IIO_HID);
421