diff options
30 files changed, 313 insertions, 210 deletions
diff --git a/drivers/iio/accel/hid-sensor-accel-3d.c b/drivers/iio/accel/hid-sensor-accel-3d.c index 314a4057879e..a95cda0e387f 100644 --- a/drivers/iio/accel/hid-sensor-accel-3d.c +++ b/drivers/iio/accel/hid-sensor-accel-3d.c | |||
@@ -197,21 +197,8 @@ static const struct iio_info accel_3d_info = { | |||
197 | /* Function to push data to buffer */ | 197 | /* Function to push data to buffer */ |
198 | static void hid_sensor_push_data(struct iio_dev *indio_dev, u8 *data, int len) | 198 | static void hid_sensor_push_data(struct iio_dev *indio_dev, u8 *data, int len) |
199 | { | 199 | { |
200 | struct iio_buffer *buffer = indio_dev->buffer; | ||
201 | int datum_sz; | ||
202 | |||
203 | dev_dbg(&indio_dev->dev, "hid_sensor_push_data\n"); | 200 | dev_dbg(&indio_dev->dev, "hid_sensor_push_data\n"); |
204 | if (!buffer) { | 201 | iio_push_to_buffers(indio_dev, (u8 *)data); |
205 | dev_err(&indio_dev->dev, "Buffer == NULL\n"); | ||
206 | return; | ||
207 | } | ||
208 | datum_sz = buffer->access->get_bytes_per_datum(buffer); | ||
209 | if (len > datum_sz) { | ||
210 | dev_err(&indio_dev->dev, "Datum size mismatch %d:%d\n", len, | ||
211 | datum_sz); | ||
212 | return; | ||
213 | } | ||
214 | iio_push_to_buffer(buffer, (u8 *)data); | ||
215 | } | 202 | } |
216 | 203 | ||
217 | /* Callback handler to send event after all samples are received and captured */ | 204 | /* Callback handler to send event after all samples are received and captured */ |
diff --git a/drivers/iio/adc/ad7266.c b/drivers/iio/adc/ad7266.c index b11f214779a2..a6f4fc5f8201 100644 --- a/drivers/iio/adc/ad7266.c +++ b/drivers/iio/adc/ad7266.c | |||
@@ -91,7 +91,6 @@ static irqreturn_t ad7266_trigger_handler(int irq, void *p) | |||
91 | { | 91 | { |
92 | struct iio_poll_func *pf = p; | 92 | struct iio_poll_func *pf = p; |
93 | struct iio_dev *indio_dev = pf->indio_dev; | 93 | struct iio_dev *indio_dev = pf->indio_dev; |
94 | struct iio_buffer *buffer = indio_dev->buffer; | ||
95 | struct ad7266_state *st = iio_priv(indio_dev); | 94 | struct ad7266_state *st = iio_priv(indio_dev); |
96 | int ret; | 95 | int ret; |
97 | 96 | ||
@@ -99,7 +98,7 @@ static irqreturn_t ad7266_trigger_handler(int irq, void *p) | |||
99 | if (ret == 0) { | 98 | if (ret == 0) { |
100 | if (indio_dev->scan_timestamp) | 99 | if (indio_dev->scan_timestamp) |
101 | ((s64 *)st->data)[1] = pf->timestamp; | 100 | ((s64 *)st->data)[1] = pf->timestamp; |
102 | iio_push_to_buffer(buffer, (u8 *)st->data); | 101 | iio_push_to_buffers(indio_dev, (u8 *)st->data); |
103 | } | 102 | } |
104 | 103 | ||
105 | iio_trigger_notify_done(indio_dev->trig); | 104 | iio_trigger_notify_done(indio_dev->trig); |
diff --git a/drivers/iio/adc/ad7476.c b/drivers/iio/adc/ad7476.c index 7f2f45a0a48d..330248bfebae 100644 --- a/drivers/iio/adc/ad7476.c +++ b/drivers/iio/adc/ad7476.c | |||
@@ -76,7 +76,7 @@ static irqreturn_t ad7476_trigger_handler(int irq, void *p) | |||
76 | if (indio_dev->scan_timestamp) | 76 | if (indio_dev->scan_timestamp) |
77 | ((s64 *)st->data)[1] = time_ns; | 77 | ((s64 *)st->data)[1] = time_ns; |
78 | 78 | ||
79 | iio_push_to_buffer(indio_dev->buffer, st->data); | 79 | iio_push_to_buffers(indio_dev, st->data); |
80 | done: | 80 | done: |
81 | iio_trigger_notify_done(indio_dev->trig); | 81 | iio_trigger_notify_done(indio_dev->trig); |
82 | 82 | ||
diff --git a/drivers/iio/adc/ad7887.c b/drivers/iio/adc/ad7887.c index fd62309b4d3d..81153fafac7a 100644 --- a/drivers/iio/adc/ad7887.c +++ b/drivers/iio/adc/ad7887.c | |||
@@ -134,7 +134,7 @@ static irqreturn_t ad7887_trigger_handler(int irq, void *p) | |||
134 | memcpy(st->data + indio_dev->scan_bytes - sizeof(s64), | 134 | memcpy(st->data + indio_dev->scan_bytes - sizeof(s64), |
135 | &time_ns, sizeof(time_ns)); | 135 | &time_ns, sizeof(time_ns)); |
136 | 136 | ||
137 | iio_push_to_buffer(indio_dev->buffer, st->data); | 137 | iio_push_to_buffers(indio_dev, st->data); |
138 | done: | 138 | done: |
139 | iio_trigger_notify_done(indio_dev->trig); | 139 | iio_trigger_notify_done(indio_dev->trig); |
140 | 140 | ||
diff --git a/drivers/iio/adc/ad_sigma_delta.c b/drivers/iio/adc/ad_sigma_delta.c index 67baa1363d7a..afe6d78c8ff0 100644 --- a/drivers/iio/adc/ad_sigma_delta.c +++ b/drivers/iio/adc/ad_sigma_delta.c | |||
@@ -391,7 +391,7 @@ static irqreturn_t ad_sd_trigger_handler(int irq, void *p) | |||
391 | break; | 391 | break; |
392 | } | 392 | } |
393 | 393 | ||
394 | iio_push_to_buffer(indio_dev->buffer, (uint8_t *)data); | 394 | iio_push_to_buffers(indio_dev, (uint8_t *)data); |
395 | 395 | ||
396 | iio_trigger_notify_done(indio_dev->trig); | 396 | iio_trigger_notify_done(indio_dev->trig); |
397 | sigma_delta->irq_dis = false; | 397 | sigma_delta->irq_dis = false; |
diff --git a/drivers/iio/adc/at91_adc.c b/drivers/iio/adc/at91_adc.c index 2e2c9a80aa37..03b85940f4ba 100644 --- a/drivers/iio/adc/at91_adc.c +++ b/drivers/iio/adc/at91_adc.c | |||
@@ -65,7 +65,6 @@ static irqreturn_t at91_adc_trigger_handler(int irq, void *p) | |||
65 | struct iio_poll_func *pf = p; | 65 | struct iio_poll_func *pf = p; |
66 | struct iio_dev *idev = pf->indio_dev; | 66 | struct iio_dev *idev = pf->indio_dev; |
67 | struct at91_adc_state *st = iio_priv(idev); | 67 | struct at91_adc_state *st = iio_priv(idev); |
68 | struct iio_buffer *buffer = idev->buffer; | ||
69 | int i, j = 0; | 68 | int i, j = 0; |
70 | 69 | ||
71 | for (i = 0; i < idev->masklength; i++) { | 70 | for (i = 0; i < idev->masklength; i++) { |
@@ -81,7 +80,7 @@ static irqreturn_t at91_adc_trigger_handler(int irq, void *p) | |||
81 | *timestamp = pf->timestamp; | 80 | *timestamp = pf->timestamp; |
82 | } | 81 | } |
83 | 82 | ||
84 | iio_push_to_buffer(buffer, st->buffer); | 83 | iio_push_to_buffers(indio_dev, (u8 *)st->buffer); |
85 | 84 | ||
86 | iio_trigger_notify_done(idev->trig); | 85 | iio_trigger_notify_done(idev->trig); |
87 | 86 | ||
diff --git a/drivers/iio/gyro/hid-sensor-gyro-3d.c b/drivers/iio/gyro/hid-sensor-gyro-3d.c index 4c56ada51c39..02ef989b830d 100644 --- a/drivers/iio/gyro/hid-sensor-gyro-3d.c +++ b/drivers/iio/gyro/hid-sensor-gyro-3d.c | |||
@@ -197,21 +197,8 @@ static const struct iio_info gyro_3d_info = { | |||
197 | /* Function to push data to buffer */ | 197 | /* Function to push data to buffer */ |
198 | static void hid_sensor_push_data(struct iio_dev *indio_dev, u8 *data, int len) | 198 | static void hid_sensor_push_data(struct iio_dev *indio_dev, u8 *data, int len) |
199 | { | 199 | { |
200 | struct iio_buffer *buffer = indio_dev->buffer; | ||
201 | int datum_sz; | ||
202 | |||
203 | dev_dbg(&indio_dev->dev, "hid_sensor_push_data\n"); | 200 | dev_dbg(&indio_dev->dev, "hid_sensor_push_data\n"); |
204 | if (!buffer) { | 201 | iio_push_to_buffers(indio_dev, (u8 *)data); |
205 | dev_err(&indio_dev->dev, "Buffer == NULL\n"); | ||
206 | return; | ||
207 | } | ||
208 | datum_sz = buffer->access->get_bytes_per_datum(buffer); | ||
209 | if (len > datum_sz) { | ||
210 | dev_err(&indio_dev->dev, "Datum size mismatch %d:%d\n", len, | ||
211 | datum_sz); | ||
212 | return; | ||
213 | } | ||
214 | iio_push_to_buffer(buffer, (u8 *)data); | ||
215 | } | 202 | } |
216 | 203 | ||
217 | /* Callback handler to send event after all samples are received and captured */ | 204 | /* Callback handler to send event after all samples are received and captured */ |
diff --git a/drivers/iio/industrialio-buffer.c b/drivers/iio/industrialio-buffer.c index 722a83fd8d85..aaadd32f9f0d 100644 --- a/drivers/iio/industrialio-buffer.c +++ b/drivers/iio/industrialio-buffer.c | |||
@@ -31,6 +31,18 @@ static const char * const iio_endian_prefix[] = { | |||
31 | [IIO_LE] = "le", | 31 | [IIO_LE] = "le", |
32 | }; | 32 | }; |
33 | 33 | ||
34 | static bool iio_buffer_is_active(struct iio_dev *indio_dev, | ||
35 | struct iio_buffer *buf) | ||
36 | { | ||
37 | struct list_head *p; | ||
38 | |||
39 | list_for_each(p, &indio_dev->buffer_list) | ||
40 | if (p == &buf->buffer_list) | ||
41 | return true; | ||
42 | |||
43 | return false; | ||
44 | } | ||
45 | |||
34 | /** | 46 | /** |
35 | * iio_buffer_read_first_n_outer() - chrdev read for buffer access | 47 | * iio_buffer_read_first_n_outer() - chrdev read for buffer access |
36 | * | 48 | * |
@@ -134,7 +146,7 @@ static ssize_t iio_scan_el_store(struct device *dev, | |||
134 | if (ret < 0) | 146 | if (ret < 0) |
135 | return ret; | 147 | return ret; |
136 | mutex_lock(&indio_dev->mlock); | 148 | mutex_lock(&indio_dev->mlock); |
137 | if (iio_buffer_enabled(indio_dev)) { | 149 | if (iio_buffer_is_active(indio_dev, indio_dev->buffer)) { |
138 | ret = -EBUSY; | 150 | ret = -EBUSY; |
139 | goto error_ret; | 151 | goto error_ret; |
140 | } | 152 | } |
@@ -180,12 +192,11 @@ static ssize_t iio_scan_el_ts_store(struct device *dev, | |||
180 | return ret; | 192 | return ret; |
181 | 193 | ||
182 | mutex_lock(&indio_dev->mlock); | 194 | mutex_lock(&indio_dev->mlock); |
183 | if (iio_buffer_enabled(indio_dev)) { | 195 | if (iio_buffer_is_active(indio_dev, indio_dev->buffer)) { |
184 | ret = -EBUSY; | 196 | ret = -EBUSY; |
185 | goto error_ret; | 197 | goto error_ret; |
186 | } | 198 | } |
187 | indio_dev->buffer->scan_timestamp = state; | 199 | indio_dev->buffer->scan_timestamp = state; |
188 | indio_dev->scan_timestamp = state; | ||
189 | error_ret: | 200 | error_ret: |
190 | mutex_unlock(&indio_dev->mlock); | 201 | mutex_unlock(&indio_dev->mlock); |
191 | 202 | ||
@@ -385,7 +396,7 @@ ssize_t iio_buffer_write_length(struct device *dev, | |||
385 | return len; | 396 | return len; |
386 | 397 | ||
387 | mutex_lock(&indio_dev->mlock); | 398 | mutex_lock(&indio_dev->mlock); |
388 | if (iio_buffer_enabled(indio_dev)) { | 399 | if (iio_buffer_is_active(indio_dev, indio_dev->buffer)) { |
389 | ret = -EBUSY; | 400 | ret = -EBUSY; |
390 | } else { | 401 | } else { |
391 | if (buffer->access->set_length) | 402 | if (buffer->access->set_length) |
@@ -398,102 +409,14 @@ ssize_t iio_buffer_write_length(struct device *dev, | |||
398 | } | 409 | } |
399 | EXPORT_SYMBOL(iio_buffer_write_length); | 410 | EXPORT_SYMBOL(iio_buffer_write_length); |
400 | 411 | ||
401 | ssize_t iio_buffer_store_enable(struct device *dev, | ||
402 | struct device_attribute *attr, | ||
403 | const char *buf, | ||
404 | size_t len) | ||
405 | { | ||
406 | int ret; | ||
407 | bool requested_state, current_state; | ||
408 | int previous_mode; | ||
409 | struct iio_dev *indio_dev = dev_to_iio_dev(dev); | ||
410 | struct iio_buffer *buffer = indio_dev->buffer; | ||
411 | |||
412 | mutex_lock(&indio_dev->mlock); | ||
413 | previous_mode = indio_dev->currentmode; | ||
414 | requested_state = !(buf[0] == '0'); | ||
415 | current_state = iio_buffer_enabled(indio_dev); | ||
416 | if (current_state == requested_state) { | ||
417 | printk(KERN_INFO "iio-buffer, current state requested again\n"); | ||
418 | goto done; | ||
419 | } | ||
420 | if (requested_state) { | ||
421 | if (indio_dev->setup_ops->preenable) { | ||
422 | ret = indio_dev->setup_ops->preenable(indio_dev); | ||
423 | if (ret) { | ||
424 | printk(KERN_ERR | ||
425 | "Buffer not started: " | ||
426 | "buffer preenable failed\n"); | ||
427 | goto error_ret; | ||
428 | } | ||
429 | } | ||
430 | if (buffer->access->request_update) { | ||
431 | ret = buffer->access->request_update(buffer); | ||
432 | if (ret) { | ||
433 | printk(KERN_INFO | ||
434 | "Buffer not started: " | ||
435 | "buffer parameter update failed\n"); | ||
436 | goto error_ret; | ||
437 | } | ||
438 | } | ||
439 | /* Definitely possible for devices to support both of these. */ | ||
440 | if (indio_dev->modes & INDIO_BUFFER_TRIGGERED) { | ||
441 | if (!indio_dev->trig) { | ||
442 | printk(KERN_INFO | ||
443 | "Buffer not started: no trigger\n"); | ||
444 | ret = -EINVAL; | ||
445 | goto error_ret; | ||
446 | } | ||
447 | indio_dev->currentmode = INDIO_BUFFER_TRIGGERED; | ||
448 | } else if (indio_dev->modes & INDIO_BUFFER_HARDWARE) | ||
449 | indio_dev->currentmode = INDIO_BUFFER_HARDWARE; | ||
450 | else { /* should never be reached */ | ||
451 | ret = -EINVAL; | ||
452 | goto error_ret; | ||
453 | } | ||
454 | |||
455 | if (indio_dev->setup_ops->postenable) { | ||
456 | ret = indio_dev->setup_ops->postenable(indio_dev); | ||
457 | if (ret) { | ||
458 | printk(KERN_INFO | ||
459 | "Buffer not started: " | ||
460 | "postenable failed\n"); | ||
461 | indio_dev->currentmode = previous_mode; | ||
462 | if (indio_dev->setup_ops->postdisable) | ||
463 | indio_dev->setup_ops-> | ||
464 | postdisable(indio_dev); | ||
465 | goto error_ret; | ||
466 | } | ||
467 | } | ||
468 | } else { | ||
469 | if (indio_dev->setup_ops->predisable) { | ||
470 | ret = indio_dev->setup_ops->predisable(indio_dev); | ||
471 | if (ret) | ||
472 | goto error_ret; | ||
473 | } | ||
474 | indio_dev->currentmode = INDIO_DIRECT_MODE; | ||
475 | if (indio_dev->setup_ops->postdisable) { | ||
476 | ret = indio_dev->setup_ops->postdisable(indio_dev); | ||
477 | if (ret) | ||
478 | goto error_ret; | ||
479 | } | ||
480 | } | ||
481 | done: | ||
482 | mutex_unlock(&indio_dev->mlock); | ||
483 | return len; | ||
484 | |||
485 | error_ret: | ||
486 | mutex_unlock(&indio_dev->mlock); | ||
487 | return ret; | ||
488 | } | ||
489 | EXPORT_SYMBOL(iio_buffer_store_enable); | ||
490 | |||
491 | ssize_t iio_buffer_show_enable(struct device *dev, | 412 | ssize_t iio_buffer_show_enable(struct device *dev, |
492 | struct device_attribute *attr, | 413 | struct device_attribute *attr, |
493 | char *buf) | 414 | char *buf) |
494 | { | 415 | { |
495 | struct iio_dev *indio_dev = dev_to_iio_dev(dev); | 416 | struct iio_dev *indio_dev = dev_to_iio_dev(dev); |
496 | return sprintf(buf, "%d\n", iio_buffer_enabled(indio_dev)); | 417 | return sprintf(buf, "%d\n", |
418 | iio_buffer_is_active(indio_dev, | ||
419 | indio_dev->buffer)); | ||
497 | } | 420 | } |
498 | EXPORT_SYMBOL(iio_buffer_show_enable); | 421 | EXPORT_SYMBOL(iio_buffer_show_enable); |
499 | 422 | ||
@@ -537,35 +460,220 @@ static int iio_compute_scan_bytes(struct iio_dev *indio_dev, const long *mask, | |||
537 | return bytes; | 460 | return bytes; |
538 | } | 461 | } |
539 | 462 | ||
540 | int iio_sw_buffer_preenable(struct iio_dev *indio_dev) | 463 | int iio_update_buffers(struct iio_dev *indio_dev, |
464 | struct iio_buffer *insert_buffer, | ||
465 | struct iio_buffer *remove_buffer) | ||
541 | { | 466 | { |
542 | struct iio_buffer *buffer = indio_dev->buffer; | 467 | int ret; |
543 | dev_dbg(&indio_dev->dev, "%s\n", __func__); | 468 | int success = 0; |
469 | struct iio_buffer *buffer; | ||
470 | unsigned long *compound_mask; | ||
471 | const unsigned long *old_mask; | ||
544 | 472 | ||
545 | /* How much space will the demuxed element take? */ | 473 | /* Wind down existing buffers - iff there are any */ |
546 | indio_dev->scan_bytes = | 474 | if (!list_empty(&indio_dev->buffer_list)) { |
547 | iio_compute_scan_bytes(indio_dev, buffer->scan_mask, | 475 | if (indio_dev->setup_ops->predisable) { |
548 | buffer->scan_timestamp); | 476 | ret = indio_dev->setup_ops->predisable(indio_dev); |
549 | buffer->access->set_bytes_per_datum(buffer, indio_dev->scan_bytes); | 477 | if (ret) |
478 | goto error_ret; | ||
479 | } | ||
480 | indio_dev->currentmode = INDIO_DIRECT_MODE; | ||
481 | if (indio_dev->setup_ops->postdisable) { | ||
482 | ret = indio_dev->setup_ops->postdisable(indio_dev); | ||
483 | if (ret) | ||
484 | goto error_ret; | ||
485 | } | ||
486 | } | ||
487 | /* Keep a copy of current setup to allow roll back */ | ||
488 | old_mask = indio_dev->active_scan_mask; | ||
489 | if (!indio_dev->available_scan_masks) | ||
490 | indio_dev->active_scan_mask = NULL; | ||
491 | |||
492 | if (remove_buffer) | ||
493 | list_del(&remove_buffer->buffer_list); | ||
494 | if (insert_buffer) | ||
495 | list_add(&insert_buffer->buffer_list, &indio_dev->buffer_list); | ||
496 | |||
497 | /* If no buffers in list, we are done */ | ||
498 | if (list_empty(&indio_dev->buffer_list)) { | ||
499 | indio_dev->currentmode = INDIO_DIRECT_MODE; | ||
500 | if (indio_dev->available_scan_masks == NULL) | ||
501 | kfree(old_mask); | ||
502 | return 0; | ||
503 | } | ||
550 | 504 | ||
551 | /* What scan mask do we actually have ?*/ | 505 | /* What scan mask do we actually have ?*/ |
552 | if (indio_dev->available_scan_masks) | 506 | compound_mask = kcalloc(BITS_TO_LONGS(indio_dev->masklength), |
507 | sizeof(long), GFP_KERNEL); | ||
508 | if (compound_mask == NULL) { | ||
509 | if (indio_dev->available_scan_masks == NULL) | ||
510 | kfree(old_mask); | ||
511 | return -ENOMEM; | ||
512 | } | ||
513 | indio_dev->scan_timestamp = 0; | ||
514 | |||
515 | list_for_each_entry(buffer, &indio_dev->buffer_list, buffer_list) { | ||
516 | bitmap_or(compound_mask, compound_mask, buffer->scan_mask, | ||
517 | indio_dev->masklength); | ||
518 | indio_dev->scan_timestamp |= buffer->scan_timestamp; | ||
519 | } | ||
520 | if (indio_dev->available_scan_masks) { | ||
553 | indio_dev->active_scan_mask = | 521 | indio_dev->active_scan_mask = |
554 | iio_scan_mask_match(indio_dev->available_scan_masks, | 522 | iio_scan_mask_match(indio_dev->available_scan_masks, |
555 | indio_dev->masklength, | 523 | indio_dev->masklength, |
556 | buffer->scan_mask); | 524 | compound_mask); |
557 | else | 525 | if (indio_dev->active_scan_mask == NULL) { |
558 | indio_dev->active_scan_mask = buffer->scan_mask; | 526 | /* |
559 | 527 | * Roll back. | |
560 | if (indio_dev->active_scan_mask == NULL) | 528 | * Note can only occur when adding a buffer. |
561 | return -EINVAL; | 529 | */ |
530 | list_del(&insert_buffer->buffer_list); | ||
531 | indio_dev->active_scan_mask = old_mask; | ||
532 | success = -EINVAL; | ||
533 | } | ||
534 | } else { | ||
535 | indio_dev->active_scan_mask = compound_mask; | ||
536 | } | ||
562 | 537 | ||
563 | iio_update_demux(indio_dev); | 538 | iio_update_demux(indio_dev); |
564 | 539 | ||
565 | if (indio_dev->info->update_scan_mode) | 540 | /* Wind up again */ |
566 | return indio_dev->info | 541 | if (indio_dev->setup_ops->preenable) { |
542 | ret = indio_dev->setup_ops->preenable(indio_dev); | ||
543 | if (ret) { | ||
544 | printk(KERN_ERR | ||
545 | "Buffer not started:" | ||
546 | "buffer preenable failed\n"); | ||
547 | goto error_remove_inserted; | ||
548 | } | ||
549 | } | ||
550 | indio_dev->scan_bytes = | ||
551 | iio_compute_scan_bytes(indio_dev, | ||
552 | indio_dev->active_scan_mask, | ||
553 | indio_dev->scan_timestamp); | ||
554 | list_for_each_entry(buffer, &indio_dev->buffer_list, buffer_list) | ||
555 | if (buffer->access->request_update) { | ||
556 | ret = buffer->access->request_update(buffer); | ||
557 | if (ret) { | ||
558 | printk(KERN_INFO | ||
559 | "Buffer not started:" | ||
560 | "buffer parameter update failed\n"); | ||
561 | goto error_run_postdisable; | ||
562 | } | ||
563 | } | ||
564 | if (indio_dev->info->update_scan_mode) { | ||
565 | ret = indio_dev->info | ||
567 | ->update_scan_mode(indio_dev, | 566 | ->update_scan_mode(indio_dev, |
568 | indio_dev->active_scan_mask); | 567 | indio_dev->active_scan_mask); |
568 | if (ret < 0) { | ||
569 | printk(KERN_INFO "update scan mode failed\n"); | ||
570 | goto error_run_postdisable; | ||
571 | } | ||
572 | } | ||
573 | /* Definitely possible for devices to support both of these.*/ | ||
574 | if (indio_dev->modes & INDIO_BUFFER_TRIGGERED) { | ||
575 | if (!indio_dev->trig) { | ||
576 | printk(KERN_INFO "Buffer not started: no trigger\n"); | ||
577 | ret = -EINVAL; | ||
578 | /* Can only occur on first buffer */ | ||
579 | goto error_run_postdisable; | ||
580 | } | ||
581 | indio_dev->currentmode = INDIO_BUFFER_TRIGGERED; | ||
582 | } else if (indio_dev->modes & INDIO_BUFFER_HARDWARE) { | ||
583 | indio_dev->currentmode = INDIO_BUFFER_HARDWARE; | ||
584 | } else { /* should never be reached */ | ||
585 | ret = -EINVAL; | ||
586 | goto error_run_postdisable; | ||
587 | } | ||
588 | |||
589 | if (indio_dev->setup_ops->postenable) { | ||
590 | ret = indio_dev->setup_ops->postenable(indio_dev); | ||
591 | if (ret) { | ||
592 | printk(KERN_INFO | ||
593 | "Buffer not started: postenable failed\n"); | ||
594 | indio_dev->currentmode = INDIO_DIRECT_MODE; | ||
595 | if (indio_dev->setup_ops->postdisable) | ||
596 | indio_dev->setup_ops->postdisable(indio_dev); | ||
597 | goto error_disable_all_buffers; | ||
598 | } | ||
599 | } | ||
600 | |||
601 | if (indio_dev->available_scan_masks) | ||
602 | kfree(compound_mask); | ||
603 | else | ||
604 | kfree(old_mask); | ||
605 | |||
606 | return success; | ||
607 | |||
608 | error_disable_all_buffers: | ||
609 | indio_dev->currentmode = INDIO_DIRECT_MODE; | ||
610 | error_run_postdisable: | ||
611 | if (indio_dev->setup_ops->postdisable) | ||
612 | indio_dev->setup_ops->postdisable(indio_dev); | ||
613 | error_remove_inserted: | ||
614 | |||
615 | if (insert_buffer) | ||
616 | list_del(&insert_buffer->buffer_list); | ||
617 | indio_dev->active_scan_mask = old_mask; | ||
618 | kfree(compound_mask); | ||
619 | error_ret: | ||
620 | |||
621 | return ret; | ||
622 | } | ||
623 | EXPORT_SYMBOL_GPL(iio_update_buffers); | ||
624 | |||
625 | ssize_t iio_buffer_store_enable(struct device *dev, | ||
626 | struct device_attribute *attr, | ||
627 | const char *buf, | ||
628 | size_t len) | ||
629 | { | ||
630 | int ret; | ||
631 | bool requested_state; | ||
632 | struct iio_dev *indio_dev = dev_to_iio_dev(dev); | ||
633 | struct iio_buffer *pbuf = indio_dev->buffer; | ||
634 | bool inlist; | ||
635 | |||
636 | ret = strtobool(buf, &requested_state); | ||
637 | if (ret < 0) | ||
638 | return ret; | ||
639 | |||
640 | mutex_lock(&indio_dev->mlock); | ||
641 | |||
642 | /* Find out if it is in the list */ | ||
643 | inlist = iio_buffer_is_active(indio_dev, pbuf); | ||
644 | /* Already in desired state */ | ||
645 | if (inlist == requested_state) | ||
646 | goto done; | ||
647 | |||
648 | if (requested_state) | ||
649 | ret = iio_update_buffers(indio_dev, | ||
650 | indio_dev->buffer, NULL); | ||
651 | else | ||
652 | ret = iio_update_buffers(indio_dev, | ||
653 | NULL, indio_dev->buffer); | ||
654 | |||
655 | if (ret < 0) | ||
656 | goto done; | ||
657 | done: | ||
658 | mutex_unlock(&indio_dev->mlock); | ||
659 | return (ret < 0) ? ret : len; | ||
660 | } | ||
661 | EXPORT_SYMBOL(iio_buffer_store_enable); | ||
662 | |||
663 | int iio_sw_buffer_preenable(struct iio_dev *indio_dev) | ||
664 | { | ||
665 | struct iio_buffer *buffer; | ||
666 | unsigned bytes; | ||
667 | dev_dbg(&indio_dev->dev, "%s\n", __func__); | ||
668 | |||
669 | list_for_each_entry(buffer, &indio_dev->buffer_list, buffer_list) | ||
670 | if (buffer->access->set_bytes_per_datum) { | ||
671 | bytes = iio_compute_scan_bytes(indio_dev, | ||
672 | buffer->scan_mask, | ||
673 | buffer->scan_timestamp); | ||
674 | |||
675 | buffer->access->set_bytes_per_datum(buffer, bytes); | ||
676 | } | ||
569 | return 0; | 677 | return 0; |
570 | } | 678 | } |
571 | EXPORT_SYMBOL(iio_sw_buffer_preenable); | 679 | EXPORT_SYMBOL(iio_sw_buffer_preenable); |
@@ -599,7 +707,11 @@ static bool iio_validate_scan_mask(struct iio_dev *indio_dev, | |||
599 | * iio_scan_mask_set() - set particular bit in the scan mask | 707 | * iio_scan_mask_set() - set particular bit in the scan mask |
600 | * @buffer: the buffer whose scan mask we are interested in | 708 | * @buffer: the buffer whose scan mask we are interested in |
601 | * @bit: the bit to be set. | 709 | * @bit: the bit to be set. |
602 | **/ | 710 | * |
711 | * Note that at this point we have no way of knowing what other | ||
712 | * buffers might request, hence this code only verifies that the | ||
713 | * individual buffers request is plausible. | ||
714 | */ | ||
603 | int iio_scan_mask_set(struct iio_dev *indio_dev, | 715 | int iio_scan_mask_set(struct iio_dev *indio_dev, |
604 | struct iio_buffer *buffer, int bit) | 716 | struct iio_buffer *buffer, int bit) |
605 | { | 717 | { |
@@ -682,13 +794,12 @@ static unsigned char *iio_demux(struct iio_buffer *buffer, | |||
682 | return buffer->demux_bounce; | 794 | return buffer->demux_bounce; |
683 | } | 795 | } |
684 | 796 | ||
685 | int iio_push_to_buffer(struct iio_buffer *buffer, unsigned char *data) | 797 | static int iio_push_to_buffer(struct iio_buffer *buffer, unsigned char *data) |
686 | { | 798 | { |
687 | unsigned char *dataout = iio_demux(buffer, data); | 799 | unsigned char *dataout = iio_demux(buffer, data); |
688 | 800 | ||
689 | return buffer->access->store_to(buffer, dataout); | 801 | return buffer->access->store_to(buffer, dataout); |
690 | } | 802 | } |
691 | EXPORT_SYMBOL_GPL(iio_push_to_buffer); | ||
692 | 803 | ||
693 | static void iio_buffer_demux_free(struct iio_buffer *buffer) | 804 | static void iio_buffer_demux_free(struct iio_buffer *buffer) |
694 | { | 805 | { |
@@ -699,10 +810,26 @@ static void iio_buffer_demux_free(struct iio_buffer *buffer) | |||
699 | } | 810 | } |
700 | } | 811 | } |
701 | 812 | ||
702 | int iio_update_demux(struct iio_dev *indio_dev) | 813 | |
814 | int iio_push_to_buffers(struct iio_dev *indio_dev, unsigned char *data) | ||
815 | { | ||
816 | int ret; | ||
817 | struct iio_buffer *buf; | ||
818 | |||
819 | list_for_each_entry(buf, &indio_dev->buffer_list, buffer_list) { | ||
820 | ret = iio_push_to_buffer(buf, data); | ||
821 | if (ret < 0) | ||
822 | return ret; | ||
823 | } | ||
824 | |||
825 | return 0; | ||
826 | } | ||
827 | EXPORT_SYMBOL_GPL(iio_push_to_buffers); | ||
828 | |||
829 | static int iio_buffer_update_demux(struct iio_dev *indio_dev, | ||
830 | struct iio_buffer *buffer) | ||
703 | { | 831 | { |
704 | const struct iio_chan_spec *ch; | 832 | const struct iio_chan_spec *ch; |
705 | struct iio_buffer *buffer = indio_dev->buffer; | ||
706 | int ret, in_ind = -1, out_ind, length; | 833 | int ret, in_ind = -1, out_ind, length; |
707 | unsigned in_loc = 0, out_loc = 0; | 834 | unsigned in_loc = 0, out_loc = 0; |
708 | struct iio_demux_table *p; | 835 | struct iio_demux_table *p; |
@@ -787,4 +914,23 @@ error_clear_mux_table: | |||
787 | 914 | ||
788 | return ret; | 915 | return ret; |
789 | } | 916 | } |
917 | |||
918 | int iio_update_demux(struct iio_dev *indio_dev) | ||
919 | { | ||
920 | struct iio_buffer *buffer; | ||
921 | int ret; | ||
922 | |||
923 | list_for_each_entry(buffer, &indio_dev->buffer_list, buffer_list) { | ||
924 | ret = iio_buffer_update_demux(indio_dev, buffer); | ||
925 | if (ret < 0) | ||
926 | goto error_clear_mux_table; | ||
927 | } | ||
928 | return 0; | ||
929 | |||
930 | error_clear_mux_table: | ||
931 | list_for_each_entry(buffer, &indio_dev->buffer_list, buffer_list) | ||
932 | iio_buffer_demux_free(buffer); | ||
933 | |||
934 | return ret; | ||
935 | } | ||
790 | EXPORT_SYMBOL_GPL(iio_update_demux); | 936 | EXPORT_SYMBOL_GPL(iio_update_demux); |
diff --git a/drivers/iio/industrialio-core.c b/drivers/iio/industrialio-core.c index cd700368eed0..060a4045be85 100644 --- a/drivers/iio/industrialio-core.c +++ b/drivers/iio/industrialio-core.c | |||
@@ -856,6 +856,7 @@ struct iio_dev *iio_device_alloc(int sizeof_priv) | |||
856 | return NULL; | 856 | return NULL; |
857 | } | 857 | } |
858 | dev_set_name(&dev->dev, "iio:device%d", dev->id); | 858 | dev_set_name(&dev->dev, "iio:device%d", dev->id); |
859 | INIT_LIST_HEAD(&dev->buffer_list); | ||
859 | } | 860 | } |
860 | 861 | ||
861 | return dev; | 862 | return dev; |
diff --git a/drivers/iio/light/adjd_s311.c b/drivers/iio/light/adjd_s311.c index 164b62b91a4b..36d210a06b28 100644 --- a/drivers/iio/light/adjd_s311.c +++ b/drivers/iio/light/adjd_s311.c | |||
@@ -164,7 +164,6 @@ static irqreturn_t adjd_s311_trigger_handler(int irq, void *p) | |||
164 | struct iio_poll_func *pf = p; | 164 | struct iio_poll_func *pf = p; |
165 | struct iio_dev *indio_dev = pf->indio_dev; | 165 | struct iio_dev *indio_dev = pf->indio_dev; |
166 | struct adjd_s311_data *data = iio_priv(indio_dev); | 166 | struct adjd_s311_data *data = iio_priv(indio_dev); |
167 | struct iio_buffer *buffer = indio_dev->buffer; | ||
168 | s64 time_ns = iio_get_time_ns(); | 167 | s64 time_ns = iio_get_time_ns(); |
169 | int len = 0; | 168 | int len = 0; |
170 | int i, j = 0; | 169 | int i, j = 0; |
@@ -187,7 +186,7 @@ static irqreturn_t adjd_s311_trigger_handler(int irq, void *p) | |||
187 | if (indio_dev->scan_timestamp) | 186 | if (indio_dev->scan_timestamp) |
188 | *(s64 *)((u8 *)data->buffer + ALIGN(len, sizeof(s64))) | 187 | *(s64 *)((u8 *)data->buffer + ALIGN(len, sizeof(s64))) |
189 | = time_ns; | 188 | = time_ns; |
190 | iio_push_to_buffer(buffer, (u8 *)data->buffer); | 189 | iio_push_to_buffers(indio_dev, (u8 *)data->buffer); |
191 | 190 | ||
192 | done: | 191 | done: |
193 | iio_trigger_notify_done(indio_dev->trig); | 192 | iio_trigger_notify_done(indio_dev->trig); |
diff --git a/drivers/iio/light/hid-sensor-als.c b/drivers/iio/light/hid-sensor-als.c index 96e3691e42c4..8e1f69844eea 100644 --- a/drivers/iio/light/hid-sensor-als.c +++ b/drivers/iio/light/hid-sensor-als.c | |||
@@ -176,21 +176,8 @@ static const struct iio_info als_info = { | |||
176 | /* Function to push data to buffer */ | 176 | /* Function to push data to buffer */ |
177 | static void hid_sensor_push_data(struct iio_dev *indio_dev, u8 *data, int len) | 177 | static void hid_sensor_push_data(struct iio_dev *indio_dev, u8 *data, int len) |
178 | { | 178 | { |
179 | struct iio_buffer *buffer = indio_dev->buffer; | ||
180 | int datum_sz; | ||
181 | |||
182 | dev_dbg(&indio_dev->dev, "hid_sensor_push_data\n"); | 179 | dev_dbg(&indio_dev->dev, "hid_sensor_push_data\n"); |
183 | if (!buffer) { | 180 | iio_push_to_buffers(indio_dev, (u8 *)data); |
184 | dev_err(&indio_dev->dev, "Buffer == NULL\n"); | ||
185 | return; | ||
186 | } | ||
187 | datum_sz = buffer->access->get_bytes_per_datum(buffer); | ||
188 | if (len > datum_sz) { | ||
189 | dev_err(&indio_dev->dev, "Datum size mismatch %d:%d\n", len, | ||
190 | datum_sz); | ||
191 | return; | ||
192 | } | ||
193 | iio_push_to_buffer(buffer, (u8 *)data); | ||
194 | } | 181 | } |
195 | 182 | ||
196 | /* Callback handler to send event after all samples are received and captured */ | 183 | /* Callback handler to send event after all samples are received and captured */ |
diff --git a/drivers/iio/magnetometer/hid-sensor-magn-3d.c b/drivers/iio/magnetometer/hid-sensor-magn-3d.c index c4f0d274f577..d1b5fb74b9bf 100644 --- a/drivers/iio/magnetometer/hid-sensor-magn-3d.c +++ b/drivers/iio/magnetometer/hid-sensor-magn-3d.c | |||
@@ -198,21 +198,8 @@ static const struct iio_info magn_3d_info = { | |||
198 | /* Function to push data to buffer */ | 198 | /* Function to push data to buffer */ |
199 | static void hid_sensor_push_data(struct iio_dev *indio_dev, u8 *data, int len) | 199 | static void hid_sensor_push_data(struct iio_dev *indio_dev, u8 *data, int len) |
200 | { | 200 | { |
201 | struct iio_buffer *buffer = indio_dev->buffer; | ||
202 | int datum_sz; | ||
203 | |||
204 | dev_dbg(&indio_dev->dev, "hid_sensor_push_data\n"); | 201 | dev_dbg(&indio_dev->dev, "hid_sensor_push_data\n"); |
205 | if (!buffer) { | 202 | iio_push_to_buffers(indio_dev, (u8 *)data); |
206 | dev_err(&indio_dev->dev, "Buffer == NULL\n"); | ||
207 | return; | ||
208 | } | ||
209 | datum_sz = buffer->access->get_bytes_per_datum(buffer); | ||
210 | if (len > datum_sz) { | ||
211 | dev_err(&indio_dev->dev, "Datum size mismatch %d:%d\n", len, | ||
212 | datum_sz); | ||
213 | return; | ||
214 | } | ||
215 | iio_push_to_buffer(buffer, (u8 *)data); | ||
216 | } | 203 | } |
217 | 204 | ||
218 | /* Callback handler to send event after all samples are received and captured */ | 205 | /* Callback handler to send event after all samples are received and captured */ |
diff --git a/drivers/staging/iio/accel/adis16201_ring.c b/drivers/staging/iio/accel/adis16201_ring.c index 97c09f0c26ae..e14ca60092a2 100644 --- a/drivers/staging/iio/accel/adis16201_ring.c +++ b/drivers/staging/iio/accel/adis16201_ring.c | |||
@@ -82,7 +82,7 @@ static irqreturn_t adis16201_trigger_handler(int irq, void *p) | |||
82 | if (indio_dev->scan_timestamp) | 82 | if (indio_dev->scan_timestamp) |
83 | *((s64 *)(data + ((i + 3)/4)*4)) = pf->timestamp; | 83 | *((s64 *)(data + ((i + 3)/4)*4)) = pf->timestamp; |
84 | 84 | ||
85 | iio_push_to_buffer(indio_dev->buffer, (u8 *)data); | 85 | iio_push_to_buffers(indio_dev, (u8 *)data); |
86 | 86 | ||
87 | kfree(data); | 87 | kfree(data); |
88 | done: | 88 | done: |
diff --git a/drivers/staging/iio/accel/adis16203_ring.c b/drivers/staging/iio/accel/adis16203_ring.c index 7507e1a04591..eba2e285c84a 100644 --- a/drivers/staging/iio/accel/adis16203_ring.c +++ b/drivers/staging/iio/accel/adis16203_ring.c | |||
@@ -81,7 +81,7 @@ static irqreturn_t adis16203_trigger_handler(int irq, void *p) | |||
81 | if (indio_dev->scan_timestamp) | 81 | if (indio_dev->scan_timestamp) |
82 | *((s64 *)(data + ((i + 3)/4)*4)) = pf->timestamp; | 82 | *((s64 *)(data + ((i + 3)/4)*4)) = pf->timestamp; |
83 | 83 | ||
84 | iio_push_to_buffer(indio_dev->buffer, (u8 *)data); | 84 | iio_push_to_buffers(indio_dev, (u8 *)data); |
85 | 85 | ||
86 | kfree(data); | 86 | kfree(data); |
87 | done: | 87 | done: |
diff --git a/drivers/staging/iio/accel/adis16204_ring.c b/drivers/staging/iio/accel/adis16204_ring.c index 4c976bec986b..3611a13836c9 100644 --- a/drivers/staging/iio/accel/adis16204_ring.c +++ b/drivers/staging/iio/accel/adis16204_ring.c | |||
@@ -78,7 +78,7 @@ static irqreturn_t adis16204_trigger_handler(int irq, void *p) | |||
78 | if (indio_dev->scan_timestamp) | 78 | if (indio_dev->scan_timestamp) |
79 | *((s64 *)(data + ((i + 3)/4)*4)) = pf->timestamp; | 79 | *((s64 *)(data + ((i + 3)/4)*4)) = pf->timestamp; |
80 | 80 | ||
81 | iio_push_to_buffer(indio_dev->buffer, (u8 *)data); | 81 | iio_push_to_buffers(indio_dev, (u8 *)data); |
82 | 82 | ||
83 | kfree(data); | 83 | kfree(data); |
84 | done: | 84 | done: |
diff --git a/drivers/staging/iio/accel/adis16209_ring.c b/drivers/staging/iio/accel/adis16209_ring.c index f939e29d6c82..6af9a5dbc709 100644 --- a/drivers/staging/iio/accel/adis16209_ring.c +++ b/drivers/staging/iio/accel/adis16209_ring.c | |||
@@ -78,7 +78,7 @@ static irqreturn_t adis16209_trigger_handler(int irq, void *p) | |||
78 | if (indio_dev->scan_timestamp) | 78 | if (indio_dev->scan_timestamp) |
79 | *((s64 *)(data + ((i + 3)/4)*4)) = pf->timestamp; | 79 | *((s64 *)(data + ((i + 3)/4)*4)) = pf->timestamp; |
80 | 80 | ||
81 | iio_push_to_buffer(indio_dev->buffer, (u8 *)data); | 81 | iio_push_to_buffers(indio_dev, (u8 *)data); |
82 | 82 | ||
83 | kfree(data); | 83 | kfree(data); |
84 | done: | 84 | done: |
diff --git a/drivers/staging/iio/accel/adis16240_ring.c b/drivers/staging/iio/accel/adis16240_ring.c index caff8e25e0a2..e2ac8a8c8107 100644 --- a/drivers/staging/iio/accel/adis16240_ring.c +++ b/drivers/staging/iio/accel/adis16240_ring.c | |||
@@ -76,7 +76,7 @@ static irqreturn_t adis16240_trigger_handler(int irq, void *p) | |||
76 | if (indio_dev->scan_timestamp) | 76 | if (indio_dev->scan_timestamp) |
77 | *((s64 *)(data + ((i + 3)/4)*4)) = pf->timestamp; | 77 | *((s64 *)(data + ((i + 3)/4)*4)) = pf->timestamp; |
78 | 78 | ||
79 | iio_push_to_buffer(indio_dev->buffer, (u8 *)data); | 79 | iio_push_to_buffers(indio_dev, (u8 *)data); |
80 | 80 | ||
81 | kfree(data); | 81 | kfree(data); |
82 | done: | 82 | done: |
diff --git a/drivers/staging/iio/accel/lis3l02dq_ring.c b/drivers/staging/iio/accel/lis3l02dq_ring.c index 246352716537..bc38651c315e 100644 --- a/drivers/staging/iio/accel/lis3l02dq_ring.c +++ b/drivers/staging/iio/accel/lis3l02dq_ring.c | |||
@@ -154,7 +154,7 @@ static irqreturn_t lis3l02dq_trigger_handler(int irq, void *p) | |||
154 | if (indio_dev->scan_timestamp) | 154 | if (indio_dev->scan_timestamp) |
155 | *(s64 *)((u8 *)data + ALIGN(len, sizeof(s64))) | 155 | *(s64 *)((u8 *)data + ALIGN(len, sizeof(s64))) |
156 | = pf->timestamp; | 156 | = pf->timestamp; |
157 | iio_push_to_buffer(indio_dev->buffer, (u8 *)data); | 157 | iio_push_to_buffers(indio_dev, (u8 *)data); |
158 | 158 | ||
159 | kfree(data); | 159 | kfree(data); |
160 | done: | 160 | done: |
diff --git a/drivers/staging/iio/adc/ad7298_ring.c b/drivers/staging/iio/adc/ad7298_ring.c index c2906a85fedb..b3dd514b9627 100644 --- a/drivers/staging/iio/adc/ad7298_ring.c +++ b/drivers/staging/iio/adc/ad7298_ring.c | |||
@@ -93,7 +93,7 @@ static irqreturn_t ad7298_trigger_handler(int irq, void *p) | |||
93 | indio_dev->masklength); i++) | 93 | indio_dev->masklength); i++) |
94 | buf[i] = be16_to_cpu(st->rx_buf[i]); | 94 | buf[i] = be16_to_cpu(st->rx_buf[i]); |
95 | 95 | ||
96 | iio_push_to_buffer(indio_dev->buffer, (u8 *)buf); | 96 | iio_push_to_buffers(indio_dev, (u8 *)buf); |
97 | 97 | ||
98 | done: | 98 | done: |
99 | iio_trigger_notify_done(indio_dev->trig); | 99 | iio_trigger_notify_done(indio_dev->trig); |
diff --git a/drivers/staging/iio/adc/ad7606_ring.c b/drivers/staging/iio/adc/ad7606_ring.c index ba04d0ffd4f4..2b25cb07fe41 100644 --- a/drivers/staging/iio/adc/ad7606_ring.c +++ b/drivers/staging/iio/adc/ad7606_ring.c | |||
@@ -83,7 +83,7 @@ static void ad7606_poll_bh_to_ring(struct work_struct *work_s) | |||
83 | if (indio_dev->scan_timestamp) | 83 | if (indio_dev->scan_timestamp) |
84 | *((s64 *)(buf + indio_dev->scan_bytes - sizeof(s64))) = time_ns; | 84 | *((s64 *)(buf + indio_dev->scan_bytes - sizeof(s64))) = time_ns; |
85 | 85 | ||
86 | iio_push_to_buffer(indio_dev->buffer, buf); | 86 | iio_push_to_buffers(indio_dev, buf); |
87 | done: | 87 | done: |
88 | gpio_set_value(st->pdata->gpio_convst, 0); | 88 | gpio_set_value(st->pdata->gpio_convst, 0); |
89 | iio_trigger_notify_done(indio_dev->trig); | 89 | iio_trigger_notify_done(indio_dev->trig); |
diff --git a/drivers/staging/iio/adc/ad799x_ring.c b/drivers/staging/iio/adc/ad799x_ring.c index 86026d9b20bc..2c5f38475a8e 100644 --- a/drivers/staging/iio/adc/ad799x_ring.c +++ b/drivers/staging/iio/adc/ad799x_ring.c | |||
@@ -77,7 +77,7 @@ static irqreturn_t ad799x_trigger_handler(int irq, void *p) | |||
77 | memcpy(rxbuf + indio_dev->scan_bytes - sizeof(s64), | 77 | memcpy(rxbuf + indio_dev->scan_bytes - sizeof(s64), |
78 | &time_ns, sizeof(time_ns)); | 78 | &time_ns, sizeof(time_ns)); |
79 | 79 | ||
80 | iio_push_to_buffer(indio_dev->buffer, rxbuf); | 80 | iio_push_to_buffers(indio_dev, rxbuf); |
81 | done: | 81 | done: |
82 | kfree(rxbuf); | 82 | kfree(rxbuf); |
83 | out: | 83 | out: |
diff --git a/drivers/staging/iio/adc/max1363_ring.c b/drivers/staging/iio/adc/max1363_ring.c index 5f74f3b7671a..688304bdbaf0 100644 --- a/drivers/staging/iio/adc/max1363_ring.c +++ b/drivers/staging/iio/adc/max1363_ring.c | |||
@@ -80,7 +80,7 @@ static irqreturn_t max1363_trigger_handler(int irq, void *p) | |||
80 | 80 | ||
81 | if (indio_dev->scan_timestamp) | 81 | if (indio_dev->scan_timestamp) |
82 | memcpy(rxbuf + d_size - sizeof(s64), &time_ns, sizeof(time_ns)); | 82 | memcpy(rxbuf + d_size - sizeof(s64), &time_ns, sizeof(time_ns)); |
83 | iio_push_to_buffer(indio_dev->buffer, rxbuf); | 83 | iio_push_to_buffers(indio_dev, rxbuf); |
84 | 84 | ||
85 | done_free: | 85 | done_free: |
86 | kfree(rxbuf); | 86 | kfree(rxbuf); |
diff --git a/drivers/staging/iio/adc/mxs-lradc.c b/drivers/staging/iio/adc/mxs-lradc.c index df5bba284b73..3b467d8c5882 100644 --- a/drivers/staging/iio/adc/mxs-lradc.c +++ b/drivers/staging/iio/adc/mxs-lradc.c | |||
@@ -237,7 +237,6 @@ static irqreturn_t mxs_lradc_trigger_handler(int irq, void *p) | |||
237 | struct iio_poll_func *pf = p; | 237 | struct iio_poll_func *pf = p; |
238 | struct iio_dev *iio = pf->indio_dev; | 238 | struct iio_dev *iio = pf->indio_dev; |
239 | struct mxs_lradc *lradc = iio_priv(iio); | 239 | struct mxs_lradc *lradc = iio_priv(iio); |
240 | struct iio_buffer *buffer = iio->buffer; | ||
241 | const uint32_t chan_value = LRADC_CH_ACCUMULATE | | 240 | const uint32_t chan_value = LRADC_CH_ACCUMULATE | |
242 | ((LRADC_DELAY_TIMER_LOOP - 1) << LRADC_CH_NUM_SAMPLES_OFFSET); | 241 | ((LRADC_DELAY_TIMER_LOOP - 1) << LRADC_CH_NUM_SAMPLES_OFFSET); |
243 | int i, j = 0; | 242 | int i, j = 0; |
@@ -256,7 +255,7 @@ static irqreturn_t mxs_lradc_trigger_handler(int irq, void *p) | |||
256 | *timestamp = pf->timestamp; | 255 | *timestamp = pf->timestamp; |
257 | } | 256 | } |
258 | 257 | ||
259 | iio_push_to_buffer(buffer, (u8 *)lradc->buffer); | 258 | iio_push_to_buffers(iio, (u8 *)lradc->buffer); |
260 | 259 | ||
261 | iio_trigger_notify_done(iio->trig); | 260 | iio_trigger_notify_done(iio->trig); |
262 | 261 | ||
diff --git a/drivers/staging/iio/gyro/adis16260_ring.c b/drivers/staging/iio/gyro/adis16260_ring.c index e294cb49736d..d6c48f850a9e 100644 --- a/drivers/staging/iio/gyro/adis16260_ring.c +++ b/drivers/staging/iio/gyro/adis16260_ring.c | |||
@@ -81,7 +81,7 @@ static irqreturn_t adis16260_trigger_handler(int irq, void *p) | |||
81 | if (indio_dev->scan_timestamp) | 81 | if (indio_dev->scan_timestamp) |
82 | *((s64 *)(data + ((i + 3)/4)*4)) = pf->timestamp; | 82 | *((s64 *)(data + ((i + 3)/4)*4)) = pf->timestamp; |
83 | 83 | ||
84 | iio_push_to_buffer(indio_dev->buffer, (u8 *)data); | 84 | iio_push_to_buffers(indio_dev, (u8 *)data); |
85 | 85 | ||
86 | kfree(data); | 86 | kfree(data); |
87 | done: | 87 | done: |
diff --git a/drivers/staging/iio/iio_simple_dummy_buffer.c b/drivers/staging/iio/iio_simple_dummy_buffer.c index 697d9700db2f..dee16f0e7570 100644 --- a/drivers/staging/iio/iio_simple_dummy_buffer.c +++ b/drivers/staging/iio/iio_simple_dummy_buffer.c | |||
@@ -46,7 +46,6 @@ static irqreturn_t iio_simple_dummy_trigger_h(int irq, void *p) | |||
46 | { | 46 | { |
47 | struct iio_poll_func *pf = p; | 47 | struct iio_poll_func *pf = p; |
48 | struct iio_dev *indio_dev = pf->indio_dev; | 48 | struct iio_dev *indio_dev = pf->indio_dev; |
49 | struct iio_buffer *buffer = indio_dev->buffer; | ||
50 | int len = 0; | 49 | int len = 0; |
51 | u16 *data; | 50 | u16 *data; |
52 | 51 | ||
@@ -76,7 +75,7 @@ static irqreturn_t iio_simple_dummy_trigger_h(int irq, void *p) | |||
76 | i < bitmap_weight(indio_dev->active_scan_mask, | 75 | i < bitmap_weight(indio_dev->active_scan_mask, |
77 | indio_dev->masklength); | 76 | indio_dev->masklength); |
78 | i++, j++) { | 77 | i++, j++) { |
79 | j = find_next_bit(buffer->scan_mask, | 78 | j = find_next_bit(indio_dev->active_scan_mask, |
80 | indio_dev->masklength, j); | 79 | indio_dev->masklength, j); |
81 | /* random access read from the 'device' */ | 80 | /* random access read from the 'device' */ |
82 | data[i] = fakedata[j]; | 81 | data[i] = fakedata[j]; |
@@ -87,7 +86,7 @@ static irqreturn_t iio_simple_dummy_trigger_h(int irq, void *p) | |||
87 | if (indio_dev->scan_timestamp) | 86 | if (indio_dev->scan_timestamp) |
88 | *(s64 *)((u8 *)data + ALIGN(len, sizeof(s64))) | 87 | *(s64 *)((u8 *)data + ALIGN(len, sizeof(s64))) |
89 | = iio_get_time_ns(); | 88 | = iio_get_time_ns(); |
90 | iio_push_to_buffer(buffer, (u8 *)data); | 89 | iio_push_to_buffers(indio_dev, (u8 *)data); |
91 | 90 | ||
92 | kfree(data); | 91 | kfree(data); |
93 | 92 | ||
diff --git a/drivers/staging/iio/impedance-analyzer/ad5933.c b/drivers/staging/iio/impedance-analyzer/ad5933.c index de21d47f33e9..b1fef147d5b5 100644 --- a/drivers/staging/iio/impedance-analyzer/ad5933.c +++ b/drivers/staging/iio/impedance-analyzer/ad5933.c | |||
@@ -647,7 +647,6 @@ static void ad5933_work(struct work_struct *work) | |||
647 | struct ad5933_state *st = container_of(work, | 647 | struct ad5933_state *st = container_of(work, |
648 | struct ad5933_state, work.work); | 648 | struct ad5933_state, work.work); |
649 | struct iio_dev *indio_dev = i2c_get_clientdata(st->client); | 649 | struct iio_dev *indio_dev = i2c_get_clientdata(st->client); |
650 | struct iio_buffer *ring = indio_dev->buffer; | ||
651 | signed short buf[2]; | 650 | signed short buf[2]; |
652 | unsigned char status; | 651 | unsigned char status; |
653 | 652 | ||
@@ -677,8 +676,7 @@ static void ad5933_work(struct work_struct *work) | |||
677 | } else { | 676 | } else { |
678 | buf[0] = be16_to_cpu(buf[0]); | 677 | buf[0] = be16_to_cpu(buf[0]); |
679 | } | 678 | } |
680 | /* save datum to the ring */ | 679 | iio_push_to_buffers(indio_dev, (u8 *)buf); |
681 | iio_push_to_buffer(ring, (u8 *)buf); | ||
682 | } else { | 680 | } else { |
683 | /* no data available - try again later */ | 681 | /* no data available - try again later */ |
684 | schedule_delayed_work(&st->work, st->poll_time_jiffies); | 682 | schedule_delayed_work(&st->work, st->poll_time_jiffies); |
diff --git a/drivers/staging/iio/imu/adis16400_ring.c b/drivers/staging/iio/imu/adis16400_ring.c index 260bdd1a4681..d46c1e38cf7b 100644 --- a/drivers/staging/iio/imu/adis16400_ring.c +++ b/drivers/staging/iio/imu/adis16400_ring.c | |||
@@ -114,7 +114,6 @@ static irqreturn_t adis16400_trigger_handler(int irq, void *p) | |||
114 | struct iio_poll_func *pf = p; | 114 | struct iio_poll_func *pf = p; |
115 | struct iio_dev *indio_dev = pf->indio_dev; | 115 | struct iio_dev *indio_dev = pf->indio_dev; |
116 | struct adis16400_state *st = iio_priv(indio_dev); | 116 | struct adis16400_state *st = iio_priv(indio_dev); |
117 | struct iio_buffer *ring = indio_dev->buffer; | ||
118 | int i = 0, j, ret = 0; | 117 | int i = 0, j, ret = 0; |
119 | s16 *data; | 118 | s16 *data; |
120 | 119 | ||
@@ -148,9 +147,9 @@ static irqreturn_t adis16400_trigger_handler(int irq, void *p) | |||
148 | } | 147 | } |
149 | } | 148 | } |
150 | /* Guaranteed to be aligned with 8 byte boundary */ | 149 | /* Guaranteed to be aligned with 8 byte boundary */ |
151 | if (ring->scan_timestamp) | 150 | if (indio_dev->scan_timestamp) |
152 | *((s64 *)(data + ((i + 3)/4)*4)) = pf->timestamp; | 151 | *((s64 *)(data + ((i + 3)/4)*4)) = pf->timestamp; |
153 | iio_push_to_buffer(ring, (u8 *) data); | 152 | iio_push_to_buffers(indio_dev, (u8 *) data); |
154 | 153 | ||
155 | done: | 154 | done: |
156 | kfree(data); | 155 | kfree(data); |
diff --git a/drivers/staging/iio/meter/ade7758_ring.c b/drivers/staging/iio/meter/ade7758_ring.c index 9e49baccf660..4552a4c7fe33 100644 --- a/drivers/staging/iio/meter/ade7758_ring.c +++ b/drivers/staging/iio/meter/ade7758_ring.c | |||
@@ -73,7 +73,7 @@ static irqreturn_t ade7758_trigger_handler(int irq, void *p) | |||
73 | if (indio_dev->scan_timestamp) | 73 | if (indio_dev->scan_timestamp) |
74 | dat64[1] = pf->timestamp; | 74 | dat64[1] = pf->timestamp; |
75 | 75 | ||
76 | iio_push_to_buffer(indio_dev->buffer, (u8 *)dat64); | 76 | iio_push_to_buffers(indio_dev, (u8 *)dat64); |
77 | 77 | ||
78 | iio_trigger_notify_done(indio_dev->trig); | 78 | iio_trigger_notify_done(indio_dev->trig); |
79 | 79 | ||
diff --git a/include/linux/iio/buffer.h b/include/linux/iio/buffer.h index c629b3a1d9a9..027040569180 100644 --- a/include/linux/iio/buffer.h +++ b/include/linux/iio/buffer.h | |||
@@ -66,7 +66,8 @@ struct iio_buffer_access_funcs { | |||
66 | * @stufftoread: [INTERN] flag to indicate new data. | 66 | * @stufftoread: [INTERN] flag to indicate new data. |
67 | * @demux_list: [INTERN] list of operations required to demux the scan. | 67 | * @demux_list: [INTERN] list of operations required to demux the scan. |
68 | * @demux_bounce: [INTERN] buffer for doing gather from incoming scan. | 68 | * @demux_bounce: [INTERN] buffer for doing gather from incoming scan. |
69 | **/ | 69 | * @buffer_list: [INTERN] entry in the devices list of current buffers. |
70 | */ | ||
70 | struct iio_buffer { | 71 | struct iio_buffer { |
71 | int length; | 72 | int length; |
72 | int bytes_per_datum; | 73 | int bytes_per_datum; |
@@ -81,9 +82,22 @@ struct iio_buffer { | |||
81 | const struct attribute_group *attrs; | 82 | const struct attribute_group *attrs; |
82 | struct list_head demux_list; | 83 | struct list_head demux_list; |
83 | unsigned char *demux_bounce; | 84 | unsigned char *demux_bounce; |
85 | struct list_head buffer_list; | ||
84 | }; | 86 | }; |
85 | 87 | ||
86 | /** | 88 | /** |
89 | * iio_update_buffers() - add or remove buffer from active list | ||
90 | * @indio_dev: device to add buffer to | ||
91 | * @insert_buffer: buffer to insert | ||
92 | * @remove_buffer: buffer_to_remove | ||
93 | * | ||
94 | * Note this will tear down the all buffering and build it up again | ||
95 | */ | ||
96 | int iio_update_buffers(struct iio_dev *indio_dev, | ||
97 | struct iio_buffer *insert_buffer, | ||
98 | struct iio_buffer *remove_buffer); | ||
99 | |||
100 | /** | ||
87 | * iio_buffer_init() - Initialize the buffer structure | 101 | * iio_buffer_init() - Initialize the buffer structure |
88 | * @buffer: buffer to be initialized | 102 | * @buffer: buffer to be initialized |
89 | **/ | 103 | **/ |
@@ -115,11 +129,11 @@ int iio_scan_mask_set(struct iio_dev *indio_dev, | |||
115 | struct iio_buffer *buffer, int bit); | 129 | struct iio_buffer *buffer, int bit); |
116 | 130 | ||
117 | /** | 131 | /** |
118 | * iio_push_to_buffer() - push to a registered buffer. | 132 | * iio_push_to_buffers() - push to a registered buffer. |
119 | * @buffer: IIO buffer structure for device | 133 | * @indio_dev: iio_dev structure for device. |
120 | * @data: the data to push to the buffer | 134 | * @data: Full scan. |
121 | */ | 135 | */ |
122 | int iio_push_to_buffer(struct iio_buffer *buffer, unsigned char *data); | 136 | int iio_push_to_buffers(struct iio_dev *indio_dev, unsigned char *data); |
123 | 137 | ||
124 | int iio_update_demux(struct iio_dev *indio_dev); | 138 | int iio_update_demux(struct iio_dev *indio_dev); |
125 | 139 | ||
diff --git a/include/linux/iio/iio.h b/include/linux/iio/iio.h index 7806c24e5bc8..adca93a999a7 100644 --- a/include/linux/iio/iio.h +++ b/include/linux/iio/iio.h | |||
@@ -410,6 +410,7 @@ struct iio_buffer_setup_ops { | |||
410 | * and owner | 410 | * and owner |
411 | * @event_interface: [INTERN] event chrdevs associated with interrupt lines | 411 | * @event_interface: [INTERN] event chrdevs associated with interrupt lines |
412 | * @buffer: [DRIVER] any buffer present | 412 | * @buffer: [DRIVER] any buffer present |
413 | * @buffer_list: [INTERN] list of all buffers currently attached | ||
413 | * @scan_bytes: [INTERN] num bytes captured to be fed to buffer demux | 414 | * @scan_bytes: [INTERN] num bytes captured to be fed to buffer demux |
414 | * @mlock: [INTERN] lock used to prevent simultaneous device state | 415 | * @mlock: [INTERN] lock used to prevent simultaneous device state |
415 | * changes | 416 | * changes |
@@ -448,6 +449,7 @@ struct iio_dev { | |||
448 | struct iio_event_interface *event_interface; | 449 | struct iio_event_interface *event_interface; |
449 | 450 | ||
450 | struct iio_buffer *buffer; | 451 | struct iio_buffer *buffer; |
452 | struct list_head buffer_list; | ||
451 | int scan_bytes; | 453 | int scan_bytes; |
452 | struct mutex mlock; | 454 | struct mutex mlock; |
453 | 455 | ||