HID: hid-sensor-hub: Enhance get feature report API
authorSrinivas Pandruvada <srinivas.pandruvada@linux.intel.com>
Thu, 19 Feb 2015 23:35:25 +0000 (15:35 -0800)
committerJiri Kosina <jkosina@suse.cz>
Mon, 23 Feb 2015 14:19:49 +0000 (15:19 +0100)
Some hid sensor feature report can contain more than one reports.
This API can now support receiving multiple values from the feature
report.
Also update the parameters in the users of this API.

Signed-off-by: Srinivas Pandruvada <srinivas.pandruvada@linux.intel.com>
Acked-by: Jonathan Cameron <jic23@kernel.org>
Signed-off-by: Jiri Kosina <jkosina@suse.cz>
drivers/hid/hid-sensor-hub.c
drivers/iio/common/hid-sensors/hid-sensor-attributes.c
drivers/iio/common/hid-sensors/hid-sensor-trigger.c
include/linux/hid-sensor-hub.h

index 0a916236316413c074da93f0a6d42e58f07c7360..c025c489270d1fbd4b589acfd64ed5161dec060b 100644 (file)
@@ -223,10 +223,11 @@ done_proc:
 EXPORT_SYMBOL_GPL(sensor_hub_set_feature);
 
 int sensor_hub_get_feature(struct hid_sensor_hub_device *hsdev, u32 report_id,
-                               u32 field_index, s32 *value)
+                          u32 field_index, int buffer_size, void *buffer)
 {
        struct hid_report *report;
        struct sensor_hub_data *data = hid_get_drvdata(hsdev->hdev);
+       int report_size;
        int ret = 0;
 
        mutex_lock(&data->mutex);
@@ -238,7 +239,17 @@ int sensor_hub_get_feature(struct hid_sensor_hub_device *hsdev, u32 report_id,
        }
        hid_hw_request(hsdev->hdev, report, HID_REQ_GET_REPORT);
        hid_hw_wait(hsdev->hdev);
-       *value = report->field[field_index]->value[0];
+
+       /* calculate number of bytes required to read this field */
+       report_size = DIV_ROUND_UP(report->field[field_index]->report_size,
+                                  8) *
+                                  report->field[field_index]->report_count;
+       if (!report_size) {
+               ret = -EINVAL;
+               goto done_proc;
+       }
+       ret = min(report_size, buffer_size);
+       memcpy(buffer, report->field[field_index]->value, ret);
 
 done_proc:
        mutex_unlock(&data->mutex);
index 25b01e156d8264ced9abf9b16f397129e76fc989..e1435e98636dec2aebe55c4d20fd9f3f6a4513b4 100644 (file)
@@ -153,8 +153,8 @@ s32 hid_sensor_read_poll_value(struct hid_sensor_common *st)
        int ret;
 
        ret = sensor_hub_get_feature(st->hsdev,
-               st->poll.report_id,
-               st->poll.index, &value);
+                                    st->poll.report_id,
+                                    st->poll.index, sizeof(value), &value);
 
        if (ret < 0 || value < 0) {
                return -EINVAL;
@@ -174,8 +174,8 @@ int hid_sensor_read_samp_freq_value(struct hid_sensor_common *st,
        int ret;
 
        ret = sensor_hub_get_feature(st->hsdev,
-               st->poll.report_id,
-               st->poll.index, &value);
+                                    st->poll.report_id,
+                                    st->poll.index, sizeof(value), &value);
        if (ret < 0 || value < 0) {
                *val1 = *val2 = 0;
                return -EINVAL;
@@ -229,8 +229,9 @@ int hid_sensor_read_raw_hyst_value(struct hid_sensor_common *st,
        int ret;
 
        ret = sensor_hub_get_feature(st->hsdev,
-               st->sensitivity.report_id,
-               st->sensitivity.index, &value);
+                                    st->sensitivity.report_id,
+                                    st->sensitivity.index, sizeof(value),
+                                    &value);
        if (ret < 0 || value < 0) {
                *val1 = *val2 = 0;
                return -EINVAL;
index 92068cdbf8c7a6e1423c889f6de791cc25141071..ef0c495a8ef99d66b270a27075186f7241750fff 100644 (file)
@@ -76,8 +76,8 @@ int hid_sensor_power_state(struct hid_sensor_common *st, bool state)
        }
 
        sensor_hub_get_feature(st->hsdev, st->power_state.report_id,
-                                       st->power_state.index,
-                                       &state_val);
+                              st->power_state.index,
+                              sizeof(state_val), &state_val);
        return 0;
 }
 EXPORT_SYMBOL(hid_sensor_power_state);
index 4c49b041922d21d5e6fb9bf4bbe1e9704ba6f339..1db3320666690dc26006298c522062783d9bde84 100644 (file)
@@ -206,13 +206,15 @@ int sensor_hub_set_feature(struct hid_sensor_hub_device *hsdev, u32 report_id,
 * sensor_hub_get_feature() - Feature get request
 * @report_id:  Report id to look for
 * @field_index:        Field index inside a report
-* @value:      Place holder for return value
+* @buffer_size:        size of the buffer
+* @buffer:     buffer to copy output
 *
 * Used to get a field in feature report. For example this can get polling
-* interval, sensitivity, activate/deactivate state.
+* interval, sensitivity, activate/deactivate state. On success it returns
+* number of bytes copied to buffer. On failure, it returns value < 0.
 */
 int sensor_hub_get_feature(struct hid_sensor_hub_device *hsdev, u32 report_id,
-                       u32 field_index, s32 *value);
+                          u32 field_index, int buffer_size, void *buffer);
 
 /* hid-sensor-attributes */