Commit 29b45787 authored by Benjamin Tissoires's avatar Benjamin Tissoires Committed by Jiri Kosina

HID: i2c-hid: reorder allocation/free of buffers

Simplifies i2c_hid_alloc_buffers tests, and makes this function
responsible of the assignment of ihid->bufsize.
The condition for the reallocation in i2c_hid_start is then simpler.
Signed-off-by: default avatarBenjamin Tissoires <benjamin.tissoires@gmail.com>
Reviewed-by: default avatarJean Delvare <khali@linux-fr.org>
Signed-off-by: default avatarJiri Kosina <jkosina@suse.cz>
parent 27174cff
...@@ -464,48 +464,38 @@ static void i2c_hid_find_max_report(struct hid_device *hid, unsigned int type, ...@@ -464,48 +464,38 @@ static void i2c_hid_find_max_report(struct hid_device *hid, unsigned int type,
} }
} }
static int i2c_hid_alloc_buffers(struct i2c_hid *ihid) static void i2c_hid_free_buffers(struct i2c_hid *ihid)
{
kfree(ihid->inbuf);
kfree(ihid->argsbuf);
kfree(ihid->cmdbuf);
ihid->inbuf = NULL;
ihid->cmdbuf = NULL;
ihid->argsbuf = NULL;
ihid->bufsize = 0;
}
static int i2c_hid_alloc_buffers(struct i2c_hid *ihid, size_t report_size)
{ {
/* the worst case is computed from the set_report command with a /* the worst case is computed from the set_report command with a
* reportID > 15 and the maximum report length */ * reportID > 15 and the maximum report length */
int args_len = sizeof(__u8) + /* optional ReportID byte */ int args_len = sizeof(__u8) + /* optional ReportID byte */
sizeof(__u16) + /* data register */ sizeof(__u16) + /* data register */
sizeof(__u16) + /* size of the report */ sizeof(__u16) + /* size of the report */
ihid->bufsize; /* report */ report_size; /* report */
ihid->inbuf = kzalloc(ihid->bufsize, GFP_KERNEL);
if (!ihid->inbuf)
return -ENOMEM;
ihid->inbuf = kzalloc(report_size, GFP_KERNEL);
ihid->argsbuf = kzalloc(args_len, GFP_KERNEL); ihid->argsbuf = kzalloc(args_len, GFP_KERNEL);
if (!ihid->argsbuf) {
kfree(ihid->inbuf);
return -ENOMEM;
}
ihid->cmdbuf = kzalloc(sizeof(union command) + args_len, GFP_KERNEL); ihid->cmdbuf = kzalloc(sizeof(union command) + args_len, GFP_KERNEL);
if (!ihid->cmdbuf) { if (!ihid->inbuf || !ihid->argsbuf || !ihid->cmdbuf) {
kfree(ihid->inbuf); i2c_hid_free_buffers(ihid);
kfree(ihid->argsbuf);
ihid->inbuf = NULL;
ihid->argsbuf = NULL;
return -ENOMEM; return -ENOMEM;
} }
return 0; ihid->bufsize = report_size;
}
static void i2c_hid_free_buffers(struct i2c_hid *ihid) return 0;
{
kfree(ihid->inbuf);
kfree(ihid->argsbuf);
kfree(ihid->cmdbuf);
ihid->inbuf = NULL;
ihid->cmdbuf = NULL;
ihid->argsbuf = NULL;
} }
static int i2c_hid_get_raw_report(struct hid_device *hid, static int i2c_hid_get_raw_report(struct hid_device *hid,
...@@ -610,22 +600,19 @@ static int i2c_hid_start(struct hid_device *hid) ...@@ -610,22 +600,19 @@ static int i2c_hid_start(struct hid_device *hid)
struct i2c_client *client = hid->driver_data; struct i2c_client *client = hid->driver_data;
struct i2c_hid *ihid = i2c_get_clientdata(client); struct i2c_hid *ihid = i2c_get_clientdata(client);
int ret; int ret;
int old_bufsize = ihid->bufsize; unsigned int bufsize = HID_MIN_BUFFER_SIZE;
ihid->bufsize = HID_MIN_BUFFER_SIZE; i2c_hid_find_max_report(hid, HID_INPUT_REPORT, &bufsize);
i2c_hid_find_max_report(hid, HID_INPUT_REPORT, &ihid->bufsize); i2c_hid_find_max_report(hid, HID_OUTPUT_REPORT, &bufsize);
i2c_hid_find_max_report(hid, HID_OUTPUT_REPORT, &ihid->bufsize); i2c_hid_find_max_report(hid, HID_FEATURE_REPORT, &bufsize);
i2c_hid_find_max_report(hid, HID_FEATURE_REPORT, &ihid->bufsize);
if (ihid->bufsize > old_bufsize || !ihid->inbuf || !ihid->cmdbuf) { if (bufsize > ihid->bufsize) {
i2c_hid_free_buffers(ihid); i2c_hid_free_buffers(ihid);
ret = i2c_hid_alloc_buffers(ihid); ret = i2c_hid_alloc_buffers(ihid, bufsize);
if (ret) { if (ret)
ihid->bufsize = old_bufsize;
return ret; return ret;
}
} }
if (!(hid->quirks & HID_QUIRK_NO_INIT_REPORTS)) if (!(hid->quirks & HID_QUIRK_NO_INIT_REPORTS))
...@@ -849,8 +836,9 @@ static int __devinit i2c_hid_probe(struct i2c_client *client, ...@@ -849,8 +836,9 @@ static int __devinit i2c_hid_probe(struct i2c_client *client,
/* we need to allocate the command buffer without knowing the maximum /* we need to allocate the command buffer without knowing the maximum
* size of the reports. Let's use HID_MIN_BUFFER_SIZE, then we do the * size of the reports. Let's use HID_MIN_BUFFER_SIZE, then we do the
* real computation later. */ * real computation later. */
ihid->bufsize = HID_MIN_BUFFER_SIZE; ret = i2c_hid_alloc_buffers(ihid, HID_MIN_BUFFER_SIZE);
i2c_hid_alloc_buffers(ihid); if (ret < 0)
goto err;
ret = i2c_hid_fetch_hid_descriptor(ihid); ret = i2c_hid_fetch_hid_descriptor(ihid);
if (ret < 0) if (ret < 0)
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment