Commit ebc8f4f6 authored by Greg Kroah-Hartman's avatar Greg Kroah-Hartman

Merge tag 'iio-for-5.5b' of...

Merge tag 'iio-for-5.5b' of https://git.kernel.org/pub/scm/linux/kernel/git/jic23/iio into staging-next

Second set of IIO + counter new device support, features etc for the 5.5 cycle.

Note two merge commits in here, both for immutable branches based
of 5.4-rc1.
1. Ti eqep driver because of some file moves in precursor patches.
   I suspect no one else will pull this one.
2. ab8500 refactor as changes in power supply, hwmon and mfd trees.
   This may come via numerous trees as well as IIO.

Counter subsystem related
* ti eqep
  - New device support with bindings.
  - Includes prior file move to reflect more general use of ti-pwmss.
* Counter core
  - simplify count_read and count_write callbacks + document change.
  - fix a typo in docs.

Various subsystems related
* AB8500
  - ab8500_btemp driver converted to be an IIO consumer driver.
  - ab8500_charger driver converted to be an IIO consumer driver.
  - ab8500_fg fuel gauge driver converted to be an IIO consumer driver.
  - ab8500 hwmon driver converted to be an IIO consumer driver.
  - mfd bindings augmented with the adc channels to make the above work.
  - drop original mfd driver.

New device support
* ab8500
  - new ADC driver used by the above other subystems via the IIO consumer
    interface.
* adux1020 photometric sensor
  - new driver and dt bindings.
* fxos877cq
  - new driver for this simple(ish) IMU with DT bindings.
* intel_mrfld_adc
  -  new driver for the ADC found on Intel Merrifield platforms.
* ltc2983
  - new driver for this multi-sensor type temperature interface.
    Includes complex DT bindings.
* max1027
  - support for 12 bit devices, max1227, max1229 and max1231 + add to trivial
    bindings.
* st_lsm6dsx
  - support for the LSM6DS0 6 axis MEMs sensor.
    Note different from the LSM6DSO which the driver already supports *sigh*
  - support for the LSM6DSRX 6 axis MEMs sensor.

Features and cleanups
* ad7303
  - replace use of core mlock with a local lock with cleanly defined scope.
* ad9834
  - add a check for devm_clk_get failing.
* at91-sama5d2
  - tidy up a 0 as NULL warning.
* bmp280
  - endian type tidy ups.
  - use bulk regulator ops for a small reduction in code.
  - use devm_add_action... to simplify error path handling.
* exynos
  - drop stray semicolon.
  - use devm_platform_ioremap_resource to reduce boilerplate.
* hx711
  - various tricks to improve the frequency of read out possible.
* max1027
  - debugfs support.
  - make interrupts optional.
  - reset at probe to get clean state.
  - refactors to allow addition of new device support.
* maxim thermocouple
  - drop an unneeded semicolon.
* mb1232
  - yaml binding conversion.
* mcp320x
  - tidy up an endian types in cast warning.
* meson_saradc
  - use devm_platform_ioremap_resource to reduce boilerplate.
* mpu3050
  - make a poison value explicity big endian to supress a warning.
* pulsedlight v2
  - endian type tidy ups.
* sgp30
  - drop an excess semicolon.
* sps30
  - make truncation explicit with masking to clean up a warning.
* st sensors
  - drop gpio include as none of these support gpios.
* st_lsm6dsx
  - tidy up some alignment issues.
  - refactors to allow addition of new device support.
    * allow varients of irq related reg definitions.
    * avoid accessing active-low, open-drain regs if not provided.
    * allow varients of bdu/boot and reset regs.
    * allow for enabling or disabling wakeup sources through platform
      data (seems someone still uses this).
  - enable wake-up events for LSM6DS0
  - use the drdy mask to avoid some invalid samples during initial start
    of sensor.
  - Add support to trim the timestamp.
* stm32_adc
  - kernel-doc fixes.
* stm32_dac
  - power management support.
* stmpe-adc
  - Fix endian type of local variable.
* twl4030
  - use false / true instead of 0 / 1 for booleans.
* xilinx-xadc
  - use devm_platform_ioremap_resouce to reduce boilerplate.
* zpa2326
  - reorganise buffer handling setup to be more consistent.

Fixes (mostly recent additions)
* cpcap-adc
  - Fix mising IRQF_ONESHOT that would cause warnings to be printed.
* st_lsm6dsx
  - Sanity check the read_fifo pointer is set.
  - use locked read and update functions to prevent some races.
  - avoid accessing enable_reg if not provided.
  - take a lock to prevent a race in updating the config.
  - kernel-doc fixes.
  - document wakeup-source property in dt binding.
  - fix lsm9ds1 gyro gain definitions.

* tag 'iio-for-5.5b' of https://git.kernel.org/pub/scm/linux/kernel/git/jic23/iio: (73 commits)
  dt-bindings: iio: imu: st_lsm6dsx: add lsm6dsrx device bindings
  iio: imu: st_lsm6dsx: add support to LSM6DSRX
  iio: st: Drop GPIO include
  iio: adc: hx711: optimize performance in read cycle
  iio: adc: stm32-adc: fix kernel-doc warnings
  iio: pressure: zpa2326: fix iio_triggered_buffer_postenable position
  iio: chemical: sgp30: drop excess semicolon
  iio: adc: twl4030: Use false / true instead of 0 / 1 with booleans
  dt-bindings: iio: Add ltc2983 documentation
  iio: temperature: Add support for LTC2983
  iio: pressure: bmp280: use devm action and remove labels from probe
  iio: pressure: bmp280: use bulk regulator ops
  iio: imu: Add support for the FXOS8700 IMU
  dt-bindings: iio: imu: add fxos8700 imu binding
  staging: iio: ad9834: add a check for devm_clk_get
  iio: adc: xilinx-xadc: use devm_platform_ioremap_resource
  iio: temp: maxim thermocouple: Drop unneeded semi colon.
  iio: adc: cpcap-adc: Fix missing IRQF_ONESHOT as only threaded handler.
  iio: adc: meson_saradc: use devm_platform_ioremap_resource
  iio: adc: exynos: use devm_platform_ioremap_resource
  ...
parents d5ca94a4 5a3436dc
# SPDX-License-Identifier: GPL-2.0
%YAML 1.2
---
$id: http://devicetree.org/schemas/counter/ti-eqep.yaml#
$schema: http://devicetree.org/meta-schemas/core.yaml#
title: Texas Instruments Enhanced Quadrature Encoder Pulse (eQEP) Module
maintainers:
- David Lechner <david@lechnology.com>
properties:
compatible:
const: ti,am3352-eqep
reg:
maxItems: 1
interrupts:
description: The eQEP event interrupt
maxItems: 1
clocks:
description: The clock that determines the SYSCLKOUT rate for the eQEP
peripheral.
maxItems: 1
clock-names:
const: sysclkout
required:
- compatible
- reg
- interrupts
- clocks
- clock-names
additionalProperties: false
examples:
- |
eqep0: counter@180 {
compatible = "ti,am3352-eqep";
reg = <0x180 0x80>;
clocks = <&l4ls_gclk>;
clock-names = "sysclkout";
interrupts = <79>;
};
...
* Maxim 1027/1029/1031 Analog to Digital Converter (ADC)
Required properties:
- compatible: Should be "maxim,max1027" or "maxim,max1029" or "maxim,max1031"
- reg: SPI chip select number for the device
- interrupts: IRQ line for the ADC
see: Documentation/devicetree/bindings/interrupt-controller/interrupts.txt
Recommended properties:
- spi-max-frequency: Definition as per
Documentation/devicetree/bindings/spi/spi-bus.txt
Example:
adc@0 {
compatible = "maxim,max1027";
reg = <0>;
interrupt-parent = <&gpio5>;
interrupts = <15 IRQ_TYPE_EDGE_RISING>;
spi-max-frequency = <1000000>;
};
# SPDX-License-Identifier: (GPL-2.0 OR BSD-2-Clause)
%YAML 1.2
---
$id: http://devicetree.org/schemas/iio/imu/nxp,fxos8700.yaml#
$schema: http://devicetree.org/meta-schemas/core.yaml#
title: Freescale FXOS8700 Inertial Measurement Unit
maintainers:
- Robert Jones <rjones@gateworks.com>
description: |
Accelerometer and magnetometer combo device with an i2c and SPI interface.
https://www.nxp.com/products/sensors/motion-sensors/6-axis/digital-motion-sensor-3d-accelerometer-2g-4g-8g-plus-3d-magnetometer:FXOS8700CQ
properties:
compatible:
enum:
- nxp,fxos8700
reg:
maxItems: 1
interrupts:
minItems: 1
maxItems: 2
interrupt-names:
minItems: 1
maxItems: 2
items:
enum:
- INT1
- INT2
drive-open-drain:
type: boolean
required:
- compatible
- reg
examples:
- |
#include <dt-bindings/gpio/gpio.h>
#include <dt-bindings/interrupt-controller/irq.h>
i2c0 {
#address-cells = <1>;
#size-cells = <0>;
fxos8700@1e {
compatible = "nxp,fxos8700";
reg = <0x1e>;
interrupt-parent = <&gpio2>;
interrupts = <7 IRQ_TYPE_EDGE_RISING>;
interrupt-names = "INT1";
};
};
- |
#include <dt-bindings/gpio/gpio.h>
#include <dt-bindings/interrupt-controller/irq.h>
spi0 {
#address-cells = <1>;
#size-cells = <0>;
fxos8700@0 {
compatible = "nxp,fxos8700";
reg = <0>;
spi-max-frequency = <1000000>;
interrupt-parent = <&gpio1>;
interrupts = <7 IRQ_TYPE_EDGE_RISING>;
interrupt-names = "INT2";
};
};
...@@ -14,6 +14,8 @@ Required properties: ...@@ -14,6 +14,8 @@ Required properties:
"st,lsm6ds3tr-c" "st,lsm6ds3tr-c"
"st,ism330dhcx" "st,ism330dhcx"
"st,lsm9ds1-imu" "st,lsm9ds1-imu"
"st,lsm6ds0"
"st,lsm6dsrx"
- reg: i2c address of the sensor / spi cs line - reg: i2c address of the sensor / spi cs line
Optional properties: Optional properties:
...@@ -31,6 +33,7 @@ Optional properties: ...@@ -31,6 +33,7 @@ Optional properties:
- interrupts: interrupt mapping for IRQ. It should be configured with - interrupts: interrupt mapping for IRQ. It should be configured with
flags IRQ_TYPE_LEVEL_HIGH, IRQ_TYPE_EDGE_RISING, IRQ_TYPE_LEVEL_LOW or flags IRQ_TYPE_LEVEL_HIGH, IRQ_TYPE_EDGE_RISING, IRQ_TYPE_LEVEL_LOW or
IRQ_TYPE_EDGE_FALLING. IRQ_TYPE_EDGE_FALLING.
- wakeup-source: Enables wake up of host system on event.
Refer to interrupt-controller/interrupts.txt for generic interrupt Refer to interrupt-controller/interrupts.txt for generic interrupt
client node bindings. client node bindings.
......
# SPDX-License-Identifier: (GPL-2.0 OR BSD-2-Clause)
%YAML 1.2
---
$id: http://devicetree.org/schemas/iio/light/adux1020.yaml#
$schema: http://devicetree.org/meta-schemas/core.yaml#
title: Analog Devices ADUX1020 Photometric sensor
maintainers:
- Manivannan Sadhasivam <manivannan.sadhasivam@linaro.org>
description: |
Photometric sensor over an i2c interface.
https://www.analog.com/media/en/technical-documentation/data-sheets/ADUX1020.pdf
properties:
compatible:
enum:
- adi,adux1020
reg:
maxItems: 1
interrupts:
maxItems: 1
required:
- compatible
- reg
examples:
- |
#include <dt-bindings/interrupt-controller/irq.h>
i2c {
#address-cells = <1>;
#size-cells = <0>;
adux1020@64 {
compatible = "adi,adux1020";
reg = <0x64>;
interrupt-parent = <&msmgpio>;
interrupts = <24 IRQ_TYPE_LEVEL_HIGH>;
};
};
...
* MaxBotix I2CXL-MaxSonar ultrasonic distance sensor of type mb1202,
mb1212, mb1222, mb1232, mb1242, mb7040 or mb7137 using the i2c interface
for ranging
Required properties:
- compatible: "maxbotix,mb1202",
"maxbotix,mb1212",
"maxbotix,mb1222",
"maxbotix,mb1232",
"maxbotix,mb1242",
"maxbotix,mb7040" or
"maxbotix,mb7137"
- reg: i2c address of the device, see also i2c/i2c.txt
Optional properties:
- interrupts: Interrupt used to announce the preceding reading
request has finished and that data is available.
If no interrupt is specified the device driver
falls back to wait a fixed amount of time until
data can be retrieved.
Example:
proximity@70 {
compatible = "maxbotix,mb1232";
reg = <0x70>;
interrupt-parent = <&gpio2>;
interrupts = <2 IRQ_TYPE_EDGE_FALLING>;
};
# SPDX-License-Identifier: GPL-2.0
%YAML 1.2
---
$id: http://devicetree.org/schemas/iio/proximity/maxbotix,mb1232.yaml#
$schema: http://devicetree.org/meta-schemas/core.yaml#
title: MaxBotix I2CXL-MaxSonar ultrasonic distance sensor
maintainers:
- Andreas Klinger <ak@it-klinger.de>
description: |
MaxBotix I2CXL-MaxSonar ultrasonic distance sensor of type mb1202,
mb1212, mb1222, mb1232, mb1242, mb7040 or mb7137 using the i2c interface
for ranging
Specifications about the devices can be found at:
https://www.maxbotix.com/documents/I2CXL-MaxSonar-EZ_Datasheet.pdf
properties:
compatible:
enum:
- maxbotix,mb1202
- maxbotix,mb1212
- maxbotix,mb1222
- maxbotix,mb1232
- maxbotix,mb1242
- maxbotix,mb7040
- maxbotix,mb7137
reg:
maxItems: 1
interrupts:
description:
Interrupt used to announce the preceding reading request has finished
and that data is available. If no interrupt is specified the device
driver falls back to wait a fixed amount of time until data can be
retrieved.
maxItems: 1
required:
- compatible
- reg
additionalProperties: false
examples:
- |
#include <dt-bindings/interrupt-controller/irq.h>
i2c {
#address-cells = <1>;
#size-cells = <0>;
proximity@70 {
compatible = "maxbotix,mb1232";
reg = <0x70>;
interrupt-parent = <&gpio2>;
interrupts = <2 IRQ_TYPE_EDGE_FALLING>;
};
};
...@@ -69,6 +69,18 @@ Required child device properties: ...@@ -69,6 +69,18 @@ Required child device properties:
- compatible : "stericsson,ab8500-[bm|btemp|charger|fg|gpadc|gpio|ponkey| - compatible : "stericsson,ab8500-[bm|btemp|charger|fg|gpadc|gpio|ponkey|
pwm|regulator|rtc|sysctrl|usb]"; pwm|regulator|rtc|sysctrl|usb]";
A few child devices require ADC channels from the GPADC node. Those follow the
standard bindings from iio/iio-bindings.txt and iio/adc/adc.txt
abx500-temp : io-channels "aux1" and "aux2" for measuring external
temperatures.
ab8500-fg : io-channel "main_bat_v" for measuring main battery voltage,
ab8500-btemp : io-channels "btemp_ball" and "bat_ctrl" for measuring the
battery voltage.
ab8500-charger : io-channels "main_charger_v", "main_charger_c", "vbus_v",
"usb_charger_c" for measuring voltage and current of the
different charging supplies.
Optional child device properties: Optional child device properties:
- interrupts : contains the device IRQ(s) using the 2-cell format (see above) - interrupts : contains the device IRQ(s) using the 2-cell format (see above)
- interrupt-names : contains names of IRQ resource in the order in which they were - interrupt-names : contains names of IRQ resource in the order in which they were
...@@ -102,6 +114,113 @@ ab8500 { ...@@ -102,6 +114,113 @@ ab8500 {
39 0x4>; 39 0x4>;
interrupt-names = "HW_CONV_END", "SW_CONV_END"; interrupt-names = "HW_CONV_END", "SW_CONV_END";
vddadc-supply = <&ab8500_ldo_tvout_reg>; vddadc-supply = <&ab8500_ldo_tvout_reg>;
#address-cells = <1>;
#size-cells = <0>;
#io-channel-cells = <1>;
/* GPADC channels */
bat_ctrl: channel@1 {
reg = <0x01>;
};
btemp_ball: channel@2 {
reg = <0x02>;
};
main_charger_v: channel@3 {
reg = <0x03>;
};
acc_detect1: channel@4 {
reg = <0x04>;
};
acc_detect2: channel@5 {
reg = <0x05>;
};
adc_aux1: channel@6 {
reg = <0x06>;
};
adc_aux2: channel@7 {
reg = <0x07>;
};
main_batt_v: channel@8 {
reg = <0x08>;
};
vbus_v: channel@9 {
reg = <0x09>;
};
main_charger_c: channel@a {
reg = <0x0a>;
};
usb_charger_c: channel@b {
reg = <0x0b>;
};
bk_bat_v: channel@c {
reg = <0x0c>;
};
die_temp: channel@d {
reg = <0x0d>;
};
usb_id: channel@e {
reg = <0x0e>;
};
xtal_temp: channel@12 {
reg = <0x12>;
};
vbat_true_meas: channel@13 {
reg = <0x13>;
};
bat_ctrl_and_ibat: channel@1c {
reg = <0x1c>;
};
vbat_meas_and_ibat: channel@1d {
reg = <0x1d>;
};
vbat_true_meas_and_ibat: channel@1e {
reg = <0x1e>;
};
bat_temp_and_ibat: channel@1f {
reg = <0x1f>;
};
};
ab8500_temp {
compatible = "stericsson,abx500-temp";
io-channels = <&gpadc 0x06>,
<&gpadc 0x07>;
io-channel-name = "aux1", "aux2";
};
ab8500_battery: ab8500_battery {
stericsson,battery-type = "LIPO";
thermistor-on-batctrl;
};
ab8500_fg {
compatible = "stericsson,ab8500-fg";
battery = <&ab8500_battery>;
io-channels = <&gpadc 0x08>;
io-channel-name = "main_bat_v";
};
ab8500_btemp {
compatible = "stericsson,ab8500-btemp";
battery = <&ab8500_battery>;
io-channels = <&gpadc 0x02>,
<&gpadc 0x01>;
io-channel-name = "btemp_ball",
"bat_ctrl";
};
ab8500_charger {
compatible = "stericsson,ab8500-charger";
battery = <&ab8500_battery>;
vddadc-supply = <&ab8500_ldo_tvout_reg>;
io-channels = <&gpadc 0x03>,
<&gpadc 0x0a>,
<&gpadc 0x09>,
<&gpadc 0x0b>;
io-channel-name = "main_charger_v",
"main_charger_c",
"vbus_v",
"usb_charger_c";
}; };
ab8500-usb { ab8500-usb {
......
...@@ -114,6 +114,18 @@ properties: ...@@ -114,6 +114,18 @@ properties:
- isil,isl68137 - isil,isl68137
# 5 Bit Programmable, Pulse-Width Modulator # 5 Bit Programmable, Pulse-Width Modulator
- maxim,ds1050 - maxim,ds1050
# 10-bit 8 channels 300ks/s SPI ADC with temperature sensor
- maxim,max1027
# 10-bit 12 channels 300ks/s SPI ADC with temperature sensor
- maxim,max1029
# 10-bit 16 channels 300ks/s SPI ADC with temperature sensor
- maxim,max1031
# 12-bit 8 channels 300ks/s SPI ADC with temperature sensor
- maxim,max1227
# 12-bit 12 channels 300ks/s SPI ADC with temperature sensor
- maxim,max1229
# 12-bit 16 channels 300ks/s SPI ADC with temperature sensor
- maxim,max1231
# Low-Power, 4-/12-Channel, 2-Wire Serial, 12-Bit ADCs # Low-Power, 4-/12-Channel, 2-Wire Serial, 12-Bit ADCs
- maxim,max1237 - maxim,max1237
# PECI-to-I2C translator for PECI-to-SMBus/I2C protocol conversion # PECI-to-I2C translator for PECI-to-SMBus/I2C protocol conversion
......
...@@ -7,7 +7,7 @@ Generic Counter Interface ...@@ -7,7 +7,7 @@ Generic Counter Interface
Introduction Introduction
============ ============
Counter devices are prevalent within a diverse spectrum of industries. Counter devices are prevalent among a diverse spectrum of industries.
The ubiquitous presence of these devices necessitates a common interface The ubiquitous presence of these devices necessitates a common interface
and standard of interaction and exposure. This driver API attempts to and standard of interaction and exposure. This driver API attempts to
resolve the issue of duplicate code found among existing counter device resolve the issue of duplicate code found among existing counter device
...@@ -26,23 +26,72 @@ the Generic Counter interface. ...@@ -26,23 +26,72 @@ the Generic Counter interface.
There are three core components to a counter: There are three core components to a counter:
* Count:
Count data for a set of Signals.
* Signal: * Signal:
Input data that is evaluated by the counter to determine the count Stream of data to be evaluated by the counter.
data.
* Synapse: * Synapse:
The association of a Signal with a respective Count. Association of a Signal, and evaluation trigger, with a Count.
* Count:
Accumulation of the effects of connected Synapses.
SIGNAL
------
A Signal represents a stream of data. This is the input data that is
evaluated by the counter to determine the count data; e.g. a quadrature
signal output line of a rotary encoder. Not all counter devices provide
user access to the Signal data, so exposure is optional for drivers.
When the Signal data is available for user access, the Generic Counter
interface provides the following available signal values:
* SIGNAL_LOW:
Signal line is in a low state.
* SIGNAL_HIGH:
Signal line is in a high state.
A Signal may be associated with one or more Counts.
SYNAPSE
-------
A Synapse represents the association of a Signal with a Count. Signal
data affects respective Count data, and the Synapse represents this
relationship.
The Synapse action mode specifies the Signal data condition that
triggers the respective Count's count function evaluation to update the
count data. The Generic Counter interface provides the following
available action modes:
* None:
Signal does not trigger the count function. In Pulse-Direction count
function mode, this Signal is evaluated as Direction.
* Rising Edge:
Low state transitions to high state.
* Falling Edge:
High state transitions to low state.
* Both Edges:
Any state transition.
A counter is defined as a set of input signals associated with count
data that are generated by the evaluation of the state of the associated
input signals as defined by the respective count functions. Within the
context of the Generic Counter interface, a counter consists of Counts
each associated with a set of Signals, whose respective Synapse
instances represent the count function update conditions for the
associated Counts.
A Synapse associates one Signal with one Count.
COUNT COUNT
----- -----
A Count represents the count data for a set of Signals. The Generic A Count represents the accumulation of the effects of connected
Counter interface provides the following available count data types: Synapses; i.e. the count data for a set of Signals. The Generic
Counter interface represents the count data as a natural number.
* COUNT_POSITION:
Unsigned integer value representing position.
A Count has a count function mode which represents the update behavior A Count has a count function mode which represents the update behavior
for the count data. The Generic Counter interface provides the following for the count data. The Generic Counter interface provides the following
...@@ -86,60 +135,7 @@ available count function modes: ...@@ -86,60 +135,7 @@ available count function modes:
Any state transition on either quadrature pair signals updates the Any state transition on either quadrature pair signals updates the
respective count. Quadrature encoding determines the direction. respective count. Quadrature encoding determines the direction.
A Count has a set of one or more associated Signals. A Count has a set of one or more associated Synapses.
SIGNAL
------
A Signal represents a counter input data; this is the input data that is
evaluated by the counter to determine the count data; e.g. a quadrature
signal output line of a rotary encoder. Not all counter devices provide
user access to the Signal data.
The Generic Counter interface provides the following available signal
data types for when the Signal data is available for user access:
* SIGNAL_LEVEL:
Signal line state level. The following states are possible:
- SIGNAL_LEVEL_LOW:
Signal line is in a low state.
- SIGNAL_LEVEL_HIGH:
Signal line is in a high state.
A Signal may be associated with one or more Counts.
SYNAPSE
-------
A Synapse represents the association of a Signal with a respective
Count. Signal data affects respective Count data, and the Synapse
represents this relationship.
The Synapse action mode specifies the Signal data condition which
triggers the respective Count's count function evaluation to update the
count data. The Generic Counter interface provides the following
available action modes:
* None:
Signal does not trigger the count function. In Pulse-Direction count
function mode, this Signal is evaluated as Direction.
* Rising Edge:
Low state transitions to high state.
* Falling Edge:
High state transitions to low state.
* Both Edges:
Any state transition.
A counter is defined as a set of input signals associated with count
data that are generated by the evaluation of the state of the associated
input signals as defined by the respective count functions. Within the
context of the Generic Counter interface, a counter consists of Counts
each associated with a set of Signals, whose respective Synapse
instances represent the count function update conditions for the
associated Counts.
Paradigm Paradigm
======== ========
...@@ -286,10 +282,36 @@ if device memory-managed registration is desired. ...@@ -286,10 +282,36 @@ if device memory-managed registration is desired.
Extension sysfs attributes can be created for auxiliary functionality Extension sysfs attributes can be created for auxiliary functionality
and data by passing in defined counter_device_ext, counter_count_ext, and data by passing in defined counter_device_ext, counter_count_ext,
and counter_signal_ext structures. In these cases, the and counter_signal_ext structures. In these cases, the
counter_device_ext structure is used for global configuration of the counter_device_ext structure is used for global/miscellaneous exposure
respective Counter device, while the counter_count_ext and and configuration of the respective Counter device, while the
counter_signal_ext structures allow for auxiliary exposure and counter_count_ext and counter_signal_ext structures allow for auxiliary
configuration of a specific Count or Signal respectively. exposure and configuration of a specific Count or Signal respectively.
Determining the type of extension to create is a matter of scope.
* Signal extensions are attributes that expose information/control
specific to a Signal. These types of attributes will exist under a
Signal's directory in sysfs.
For example, if you have an invert feature for a Signal, you can have
a Signal extension called "invert" that toggles that feature:
/sys/bus/counter/devices/counterX/signalY/invert
* Count extensions are attributes that expose information/control
specific to a Count. These type of attributes will exist under a
Count's directory in sysfs.
For example, if you want to pause/unpause a Count from updating, you
can have a Count extension called "enable" that toggles such:
/sys/bus/counter/devices/counterX/countY/enable
* Device extensions are attributes that expose information/control
non-specific to a particular Count or Signal. This is where you would
put your global features or other miscellanous functionality.
For example, if your device has an overtemp sensor, you can report the
chip overheated via a device extension called "error_overtemp":
/sys/bus/counter/devices/counterX/error_overtemp
Architecture Architecture
============ ============
......
...@@ -2005,6 +2005,7 @@ F: drivers/dma/ste_dma40* ...@@ -2005,6 +2005,7 @@ F: drivers/dma/ste_dma40*
F: drivers/hwspinlock/u8500_hsem.c F: drivers/hwspinlock/u8500_hsem.c
F: drivers/i2c/busses/i2c-nomadik.c F: drivers/i2c/busses/i2c-nomadik.c
F: drivers/i2c/busses/i2c-stu300.c F: drivers/i2c/busses/i2c-stu300.c
F: drivers/iio/adc/ab8500-gpadc.c
F: drivers/mfd/ab3100* F: drivers/mfd/ab3100*
F: drivers/mfd/ab8500* F: drivers/mfd/ab8500*
F: drivers/mfd/abx500* F: drivers/mfd/abx500*
...@@ -9624,6 +9625,14 @@ S: Maintained ...@@ -9624,6 +9625,14 @@ S: Maintained
F: Documentation/devicetree/bindings/iio/dac/ltc1660.txt F: Documentation/devicetree/bindings/iio/dac/ltc1660.txt
F: drivers/iio/dac/ltc1660.c F: drivers/iio/dac/ltc1660.c
LTC2983 IIO TEMPERATURE DRIVER
M: Nuno Sá <nuno.sa@analog.com>
W: http://ez.analog.com/community/linux-device-drivers
L: linux-iio@vger.kernel.org
S: Supported
F: drivers/iio/temperature/ltc2983.c
F: Documentation/devicetree/bindings/iio/temperature/adi,ltc2983.yaml
LTC4261 HARDWARE MONITOR DRIVER LTC4261 HARDWARE MONITOR DRIVER
M: Guenter Roeck <linux@roeck-us.net> M: Guenter Roeck <linux@roeck-us.net>
L: linux-hwmon@vger.kernel.org L: linux-hwmon@vger.kernel.org
...@@ -16228,6 +16237,12 @@ S: Maintained ...@@ -16228,6 +16237,12 @@ S: Maintained
F: drivers/media/platform/davinci/ F: drivers/media/platform/davinci/
F: include/media/davinci/ F: include/media/davinci/
TI ENHANCED QUADRATURE ENCODER PULSE (eQEP) DRIVER
R: David Lechner <david@lechnology.com>
L: linux-iio@vger.kernel.org
F: Documentation/devicetree/bindings/counter/ti-eqep.yaml
F: drivers/counter/ti-eqep.c
TI ETHERNET SWITCH DRIVER (CPSW) TI ETHERNET SWITCH DRIVER (CPSW)
R: Grygorii Strashko <grygorii.strashko@ti.com> R: Grygorii Strashko <grygorii.strashko@ti.com>
L: linux-omap@vger.kernel.org L: linux-omap@vger.kernel.org
......
...@@ -150,6 +150,15 @@ config TEGRA_GMI ...@@ -150,6 +150,15 @@ config TEGRA_GMI
Driver for the Tegra Generic Memory Interface bus which can be used Driver for the Tegra Generic Memory Interface bus which can be used
to attach devices such as NOR, UART, FPGA and more. to attach devices such as NOR, UART, FPGA and more.
config TI_PWMSS
bool
default y if (ARCH_OMAP2PLUS) && (PWM_TIECAP || PWM_TIEHRPWM || TI_EQEP)
help
PWM Subsystem driver support for AM33xx SOC.
PWM submodules require PWM config space access from submodule
drivers and require common parent driver support.
config TI_SYSC config TI_SYSC
bool "TI sysc interconnect target module driver" bool "TI sysc interconnect target module driver"
depends on ARCH_OMAP2PLUS depends on ARCH_OMAP2PLUS
......
...@@ -27,6 +27,7 @@ obj-$(CONFIG_SUNXI_RSB) += sunxi-rsb.o ...@@ -27,6 +27,7 @@ obj-$(CONFIG_SUNXI_RSB) += sunxi-rsb.o
obj-$(CONFIG_SIMPLE_PM_BUS) += simple-pm-bus.o obj-$(CONFIG_SIMPLE_PM_BUS) += simple-pm-bus.o
obj-$(CONFIG_TEGRA_ACONNECT) += tegra-aconnect.o obj-$(CONFIG_TEGRA_ACONNECT) += tegra-aconnect.o
obj-$(CONFIG_TEGRA_GMI) += tegra-gmi.o obj-$(CONFIG_TEGRA_GMI) += tegra-gmi.o
obj-$(CONFIG_TI_PWMSS) += ti-pwmss.o
obj-$(CONFIG_TI_SYSC) += ti-sysc.o obj-$(CONFIG_TI_SYSC) += ti-sysc.o
obj-$(CONFIG_TS_NBUS) += ts-nbus.o obj-$(CONFIG_TS_NBUS) += ts-nbus.o
obj-$(CONFIG_UNIPHIER_SYSTEM_BUS) += uniphier-system-bus.o obj-$(CONFIG_UNIPHIER_SYSTEM_BUS) += uniphier-system-bus.o
......
...@@ -562,11 +562,10 @@ static const struct iio_chan_spec quad8_channels[] = { ...@@ -562,11 +562,10 @@ static const struct iio_chan_spec quad8_channels[] = {
}; };
static int quad8_signal_read(struct counter_device *counter, static int quad8_signal_read(struct counter_device *counter,
struct counter_signal *signal, struct counter_signal_read_value *val) struct counter_signal *signal, enum counter_signal_value *val)
{ {
const struct quad8_iio *const priv = counter->priv; const struct quad8_iio *const priv = counter->priv;
unsigned int state; unsigned int state;
enum counter_signal_level level;
/* Only Index signal levels can be read */ /* Only Index signal levels can be read */
if (signal->id < 16) if (signal->id < 16)
...@@ -575,22 +574,19 @@ static int quad8_signal_read(struct counter_device *counter, ...@@ -575,22 +574,19 @@ static int quad8_signal_read(struct counter_device *counter,
state = inb(priv->base + QUAD8_REG_INDEX_INPUT_LEVELS) state = inb(priv->base + QUAD8_REG_INDEX_INPUT_LEVELS)
& BIT(signal->id - 16); & BIT(signal->id - 16);
level = (state) ? COUNTER_SIGNAL_LEVEL_HIGH : COUNTER_SIGNAL_LEVEL_LOW; *val = (state) ? COUNTER_SIGNAL_HIGH : COUNTER_SIGNAL_LOW;
counter_signal_read_value_set(val, COUNTER_SIGNAL_LEVEL, &level);
return 0; return 0;
} }
static int quad8_count_read(struct counter_device *counter, static int quad8_count_read(struct counter_device *counter,
struct counter_count *count, struct counter_count_read_value *val) struct counter_count *count, unsigned long *val)
{ {
const struct quad8_iio *const priv = counter->priv; const struct quad8_iio *const priv = counter->priv;
const int base_offset = priv->base + 2 * count->id; const int base_offset = priv->base + 2 * count->id;
unsigned int flags; unsigned int flags;
unsigned int borrow; unsigned int borrow;
unsigned int carry; unsigned int carry;
unsigned long position;
int i; int i;
flags = inb(base_offset + 1); flags = inb(base_offset + 1);
...@@ -598,36 +594,27 @@ static int quad8_count_read(struct counter_device *counter, ...@@ -598,36 +594,27 @@ static int quad8_count_read(struct counter_device *counter,
carry = !!(flags & QUAD8_FLAG_CT); carry = !!(flags & QUAD8_FLAG_CT);
/* Borrow XOR Carry effectively doubles count range */ /* Borrow XOR Carry effectively doubles count range */
position = (unsigned long)(borrow ^ carry) << 24; *val = (unsigned long)(borrow ^ carry) << 24;
/* Reset Byte Pointer; transfer Counter to Output Latch */ /* Reset Byte Pointer; transfer Counter to Output Latch */
outb(QUAD8_CTR_RLD | QUAD8_RLD_RESET_BP | QUAD8_RLD_CNTR_OUT, outb(QUAD8_CTR_RLD | QUAD8_RLD_RESET_BP | QUAD8_RLD_CNTR_OUT,
base_offset + 1); base_offset + 1);
for (i = 0; i < 3; i++) for (i = 0; i < 3; i++)
position |= (unsigned long)inb(base_offset) << (8 * i); *val |= (unsigned long)inb(base_offset) << (8 * i);
counter_count_read_value_set(val, COUNTER_COUNT_POSITION, &position);
return 0; return 0;
} }
static int quad8_count_write(struct counter_device *counter, static int quad8_count_write(struct counter_device *counter,
struct counter_count *count, struct counter_count_write_value *val) struct counter_count *count, unsigned long val)
{ {
const struct quad8_iio *const priv = counter->priv; const struct quad8_iio *const priv = counter->priv;
const int base_offset = priv->base + 2 * count->id; const int base_offset = priv->base + 2 * count->id;
int err;
unsigned long position;
int i; int i;
err = counter_count_write_value_get(&position, COUNTER_COUNT_POSITION,
val);
if (err)
return err;
/* Only 24-bit values are supported */ /* Only 24-bit values are supported */
if (position > 0xFFFFFF) if (val > 0xFFFFFF)
return -EINVAL; return -EINVAL;
/* Reset Byte Pointer */ /* Reset Byte Pointer */
...@@ -635,7 +622,7 @@ static int quad8_count_write(struct counter_device *counter, ...@@ -635,7 +622,7 @@ static int quad8_count_write(struct counter_device *counter,
/* Counter can only be set via Preset Register */ /* Counter can only be set via Preset Register */
for (i = 0; i < 3; i++) for (i = 0; i < 3; i++)
outb(position >> (8 * i), base_offset); outb(val >> (8 * i), base_offset);
/* Transfer Preset Register to Counter */ /* Transfer Preset Register to Counter */
outb(QUAD8_CTR_RLD | QUAD8_RLD_PRESET_CNTR, base_offset + 1); outb(QUAD8_CTR_RLD | QUAD8_RLD_PRESET_CNTR, base_offset + 1);
...@@ -644,9 +631,9 @@ static int quad8_count_write(struct counter_device *counter, ...@@ -644,9 +631,9 @@ static int quad8_count_write(struct counter_device *counter,
outb(QUAD8_CTR_RLD | QUAD8_RLD_RESET_BP, base_offset + 1); outb(QUAD8_CTR_RLD | QUAD8_RLD_RESET_BP, base_offset + 1);
/* Set Preset Register back to original value */ /* Set Preset Register back to original value */
position = priv->preset[count->id]; val = priv->preset[count->id];
for (i = 0; i < 3; i++) for (i = 0; i < 3; i++)
outb(position >> (8 * i), base_offset); outb(val >> (8 * i), base_offset);
/* Reset Borrow, Carry, Compare, and Sign flags */ /* Reset Borrow, Carry, Compare, and Sign flags */
outb(QUAD8_CTR_RLD | QUAD8_RLD_RESET_FLAGS, base_offset + 1); outb(QUAD8_CTR_RLD | QUAD8_RLD_RESET_FLAGS, base_offset + 1);
......
...@@ -49,6 +49,17 @@ config STM32_LPTIMER_CNT ...@@ -49,6 +49,17 @@ config STM32_LPTIMER_CNT
To compile this driver as a module, choose M here: the To compile this driver as a module, choose M here: the
module will be called stm32-lptimer-cnt. module will be called stm32-lptimer-cnt.
config TI_EQEP
tristate "TI eQEP counter driver"
depends on (SOC_AM33XX || COMPILE_TEST)
select REGMAP_MMIO
help
Select this option to enable the Texas Instruments Enhanced Quadrature
Encoder Pulse (eQEP) counter driver.
To compile this driver as a module, choose M here: the module will be
called ti-eqep.
config FTM_QUADDEC config FTM_QUADDEC
tristate "Flex Timer Module Quadrature decoder driver" tristate "Flex Timer Module Quadrature decoder driver"
depends on HAS_IOMEM && OF depends on HAS_IOMEM && OF
......
...@@ -8,4 +8,5 @@ obj-$(CONFIG_COUNTER) += counter.o ...@@ -8,4 +8,5 @@ obj-$(CONFIG_COUNTER) += counter.o
obj-$(CONFIG_104_QUAD_8) += 104-quad-8.o obj-$(CONFIG_104_QUAD_8) += 104-quad-8.o
obj-$(CONFIG_STM32_TIMER_CNT) += stm32-timer-cnt.o obj-$(CONFIG_STM32_TIMER_CNT) += stm32-timer-cnt.o
obj-$(CONFIG_STM32_LPTIMER_CNT) += stm32-lptimer-cnt.o obj-$(CONFIG_STM32_LPTIMER_CNT) += stm32-lptimer-cnt.o
obj-$(CONFIG_TI_EQEP) += ti-eqep.o
obj-$(CONFIG_FTM_QUADDEC) += ftm-quaddec.o obj-$(CONFIG_FTM_QUADDEC) += ftm-quaddec.o
...@@ -220,86 +220,6 @@ ssize_t counter_device_enum_available_read(struct counter_device *counter, ...@@ -220,86 +220,6 @@ ssize_t counter_device_enum_available_read(struct counter_device *counter,
} }
EXPORT_SYMBOL_GPL(counter_device_enum_available_read); EXPORT_SYMBOL_GPL(counter_device_enum_available_read);
static const char *const counter_signal_level_str[] = {
[COUNTER_SIGNAL_LEVEL_LOW] = "low",
[COUNTER_SIGNAL_LEVEL_HIGH] = "high"
};
/**
* counter_signal_read_value_set - set counter_signal_read_value data
* @val: counter_signal_read_value structure to set
* @type: property Signal data represents
* @data: Signal data
*
* This function sets an opaque counter_signal_read_value structure with the
* provided Signal data.
*/
void counter_signal_read_value_set(struct counter_signal_read_value *const val,
const enum counter_signal_value_type type,
void *const data)
{
if (type == COUNTER_SIGNAL_LEVEL)
val->len = sprintf(val->buf, "%s\n",
counter_signal_level_str[*(enum counter_signal_level *)data]);
else
val->len = 0;
}
EXPORT_SYMBOL_GPL(counter_signal_read_value_set);
/**
* counter_count_read_value_set - set counter_count_read_value data
* @val: counter_count_read_value structure to set
* @type: property Count data represents
* @data: Count data
*
* This function sets an opaque counter_count_read_value structure with the
* provided Count data.
*/
void counter_count_read_value_set(struct counter_count_read_value *const val,
const enum counter_count_value_type type,
void *const data)
{
switch (type) {
case COUNTER_COUNT_POSITION:
val->len = sprintf(val->buf, "%lu\n", *(unsigned long *)data);
break;
default:
val->len = 0;
}
}
EXPORT_SYMBOL_GPL(counter_count_read_value_set);
/**
* counter_count_write_value_get - get counter_count_write_value data
* @data: Count data
* @type: property Count data represents
* @val: counter_count_write_value structure containing data
*
* This function extracts Count data from the provided opaque
* counter_count_write_value structure and stores it at the address provided by
* @data.
*
* RETURNS:
* 0 on success, negative error number on failure.
*/
int counter_count_write_value_get(void *const data,
const enum counter_count_value_type type,
const struct counter_count_write_value *const val)
{
int err;
switch (type) {
case COUNTER_COUNT_POSITION:
err = kstrtoul(val->buf, 0, data);
if (err)
return err;
break;
}
return 0;
}
EXPORT_SYMBOL_GPL(counter_count_write_value_get);
struct counter_attr_parm { struct counter_attr_parm {
struct counter_device_attr_group *group; struct counter_device_attr_group *group;
const char *prefix; const char *prefix;
...@@ -369,6 +289,11 @@ struct counter_signal_unit { ...@@ -369,6 +289,11 @@ struct counter_signal_unit {
struct counter_signal *signal; struct counter_signal *signal;
}; };
static const char *const counter_signal_value_str[] = {
[COUNTER_SIGNAL_LOW] = "low",
[COUNTER_SIGNAL_HIGH] = "high"
};
static ssize_t counter_signal_show(struct device *dev, static ssize_t counter_signal_show(struct device *dev,
struct device_attribute *attr, char *buf) struct device_attribute *attr, char *buf)
{ {
...@@ -377,13 +302,13 @@ static ssize_t counter_signal_show(struct device *dev, ...@@ -377,13 +302,13 @@ static ssize_t counter_signal_show(struct device *dev,
const struct counter_signal_unit *const component = devattr->component; const struct counter_signal_unit *const component = devattr->component;
struct counter_signal *const signal = component->signal; struct counter_signal *const signal = component->signal;
int err; int err;
struct counter_signal_read_value val = { .buf = buf }; enum counter_signal_value val;
err = counter->ops->signal_read(counter, signal, &val); err = counter->ops->signal_read(counter, signal, &val);
if (err) if (err)
return err; return err;
return val.len; return sprintf(buf, "%s\n", counter_signal_value_str[val]);
} }
struct counter_name_unit { struct counter_name_unit {
...@@ -788,13 +713,13 @@ static ssize_t counter_count_show(struct device *dev, ...@@ -788,13 +713,13 @@ static ssize_t counter_count_show(struct device *dev,
const struct counter_count_unit *const component = devattr->component; const struct counter_count_unit *const component = devattr->component;
struct counter_count *const count = component->count; struct counter_count *const count = component->count;
int err; int err;
struct counter_count_read_value val = { .buf = buf }; unsigned long val;
err = counter->ops->count_read(counter, count, &val); err = counter->ops->count_read(counter, count, &val);
if (err) if (err)
return err; return err;
return val.len; return sprintf(buf, "%lu\n", val);
} }
static ssize_t counter_count_store(struct device *dev, static ssize_t counter_count_store(struct device *dev,
...@@ -806,9 +731,13 @@ static ssize_t counter_count_store(struct device *dev, ...@@ -806,9 +731,13 @@ static ssize_t counter_count_store(struct device *dev,
const struct counter_count_unit *const component = devattr->component; const struct counter_count_unit *const component = devattr->component;
struct counter_count *const count = component->count; struct counter_count *const count = component->count;
int err; int err;
struct counter_count_write_value val = { .buf = buf }; unsigned long val;
err = kstrtoul(buf, 0, &val);
if (err)
return err;
err = counter->ops->count_write(counter, count, &val); err = counter->ops->count_write(counter, count, val);
if (err) if (err)
return err; return err;
......
...@@ -178,31 +178,25 @@ static const enum counter_count_function ftm_quaddec_count_functions[] = { ...@@ -178,31 +178,25 @@ static const enum counter_count_function ftm_quaddec_count_functions[] = {
static int ftm_quaddec_count_read(struct counter_device *counter, static int ftm_quaddec_count_read(struct counter_device *counter,
struct counter_count *count, struct counter_count *count,
struct counter_count_read_value *val) unsigned long *val)
{ {
struct ftm_quaddec *const ftm = counter->priv; struct ftm_quaddec *const ftm = counter->priv;
uint32_t cntval; uint32_t cntval;
ftm_read(ftm, FTM_CNT, &cntval); ftm_read(ftm, FTM_CNT, &cntval);
counter_count_read_value_set(val, COUNTER_COUNT_POSITION, &cntval); *val = cntval;
return 0; return 0;
} }
static int ftm_quaddec_count_write(struct counter_device *counter, static int ftm_quaddec_count_write(struct counter_device *counter,
struct counter_count *count, struct counter_count *count,
struct counter_count_write_value *val) const unsigned long val)
{ {
struct ftm_quaddec *const ftm = counter->priv; struct ftm_quaddec *const ftm = counter->priv;
u32 cnt;
int err;
err = counter_count_write_value_get(&cnt, COUNTER_COUNT_POSITION, val); if (val != 0) {
if (err)
return err;
if (cnt != 0) {
dev_warn(&ftm->pdev->dev, "Can only accept '0' as new counter value\n"); dev_warn(&ftm->pdev->dev, "Can only accept '0' as new counter value\n");
return -EINVAL; return -EINVAL;
} }
......
...@@ -377,8 +377,7 @@ static enum counter_synapse_action stm32_lptim_cnt_synapse_actions[] = { ...@@ -377,8 +377,7 @@ static enum counter_synapse_action stm32_lptim_cnt_synapse_actions[] = {
}; };
static int stm32_lptim_cnt_read(struct counter_device *counter, static int stm32_lptim_cnt_read(struct counter_device *counter,
struct counter_count *count, struct counter_count *count, unsigned long *val)
struct counter_count_read_value *val)
{ {
struct stm32_lptim_cnt *const priv = counter->priv; struct stm32_lptim_cnt *const priv = counter->priv;
u32 cnt; u32 cnt;
...@@ -388,7 +387,7 @@ static int stm32_lptim_cnt_read(struct counter_device *counter, ...@@ -388,7 +387,7 @@ static int stm32_lptim_cnt_read(struct counter_device *counter,
if (ret) if (ret)
return ret; return ret;
counter_count_read_value_set(val, COUNTER_COUNT_POSITION, &cnt); *val = cnt;
return 0; return 0;
} }
......
...@@ -48,34 +48,27 @@ static enum counter_count_function stm32_count_functions[] = { ...@@ -48,34 +48,27 @@ static enum counter_count_function stm32_count_functions[] = {
}; };
static int stm32_count_read(struct counter_device *counter, static int stm32_count_read(struct counter_device *counter,
struct counter_count *count, struct counter_count *count, unsigned long *val)
struct counter_count_read_value *val)
{ {
struct stm32_timer_cnt *const priv = counter->priv; struct stm32_timer_cnt *const priv = counter->priv;
u32 cnt; u32 cnt;
regmap_read(priv->regmap, TIM_CNT, &cnt); regmap_read(priv->regmap, TIM_CNT, &cnt);
counter_count_read_value_set(val, COUNTER_COUNT_POSITION, &cnt); *val = cnt;
return 0; return 0;
} }
static int stm32_count_write(struct counter_device *counter, static int stm32_count_write(struct counter_device *counter,
struct counter_count *count, struct counter_count *count,
struct counter_count_write_value *val) const unsigned long val)
{ {
struct stm32_timer_cnt *const priv = counter->priv; struct stm32_timer_cnt *const priv = counter->priv;
u32 cnt;
int err;
err = counter_count_write_value_get(&cnt, COUNTER_COUNT_POSITION, val);
if (err)
return err;
if (cnt > priv->ceiling) if (val > priv->ceiling)
return -EINVAL; return -EINVAL;
return regmap_write(priv->regmap, TIM_CNT, cnt); return regmap_write(priv->regmap, TIM_CNT, val);
} }
static int stm32_count_function_get(struct counter_device *counter, static int stm32_count_function_get(struct counter_device *counter,
......
This diff is collapsed.
...@@ -40,7 +40,8 @@ comment "Native drivers" ...@@ -40,7 +40,8 @@ comment "Native drivers"
config SENSORS_AB8500 config SENSORS_AB8500
tristate "AB8500 thermal monitoring" tristate "AB8500 thermal monitoring"
depends on AB8500_GPADC && AB8500_BM depends on AB8500_GPADC && AB8500_BM && (IIO = y)
default n
help help
If you say yes here you get support for the thermal sensor part If you say yes here you get support for the thermal sensor part
of the AB8500 chip. The driver includes thermal management for of the AB8500 chip. The driver includes thermal management for
......
...@@ -17,19 +17,23 @@ ...@@ -17,19 +17,23 @@
#include <linux/hwmon-sysfs.h> #include <linux/hwmon-sysfs.h>
#include <linux/mfd/abx500.h> #include <linux/mfd/abx500.h>
#include <linux/mfd/abx500/ab8500-bm.h> #include <linux/mfd/abx500/ab8500-bm.h>
#include <linux/mfd/abx500/ab8500-gpadc.h>
#include <linux/module.h> #include <linux/module.h>
#include <linux/platform_device.h> #include <linux/platform_device.h>
#include <linux/power/ab8500.h> #include <linux/power/ab8500.h>
#include <linux/reboot.h> #include <linux/reboot.h>
#include <linux/slab.h> #include <linux/slab.h>
#include <linux/sysfs.h> #include <linux/sysfs.h>
#include <linux/iio/consumer.h>
#include "abx500.h" #include "abx500.h"
#define DEFAULT_POWER_OFF_DELAY (HZ * 10) #define DEFAULT_POWER_OFF_DELAY (HZ * 10)
#define THERMAL_VCC 1800 #define THERMAL_VCC 1800
#define PULL_UP_RESISTOR 47000 #define PULL_UP_RESISTOR 47000
/* Number of monitored sensors should not greater than NUM_SENSORS */
#define AB8500_SENSOR_AUX1 0
#define AB8500_SENSOR_AUX2 1
#define AB8500_SENSOR_BTEMP_BALL 2
#define AB8500_SENSOR_BAT_CTRL 3
#define NUM_MONITORED_SENSORS 4 #define NUM_MONITORED_SENSORS 4
struct ab8500_gpadc_cfg { struct ab8500_gpadc_cfg {
...@@ -40,7 +44,8 @@ struct ab8500_gpadc_cfg { ...@@ -40,7 +44,8 @@ struct ab8500_gpadc_cfg {
}; };
struct ab8500_temp { struct ab8500_temp {
struct ab8500_gpadc *gpadc; struct iio_channel *aux1;
struct iio_channel *aux2;
struct ab8500_btemp *btemp; struct ab8500_btemp *btemp;
struct delayed_work power_off_work; struct delayed_work power_off_work;
struct ab8500_gpadc_cfg cfg; struct ab8500_gpadc_cfg cfg;
...@@ -82,15 +87,21 @@ static int ab8500_read_sensor(struct abx500_temp *data, u8 sensor, int *temp) ...@@ -82,15 +87,21 @@ static int ab8500_read_sensor(struct abx500_temp *data, u8 sensor, int *temp)
int voltage, ret; int voltage, ret;
struct ab8500_temp *ab8500_data = data->plat_data; struct ab8500_temp *ab8500_data = data->plat_data;
if (sensor == BAT_CTRL) { if (sensor == AB8500_SENSOR_BTEMP_BALL) {
*temp = ab8500_btemp_get_batctrl_temp(ab8500_data->btemp);
} else if (sensor == BTEMP_BALL) {
*temp = ab8500_btemp_get_temp(ab8500_data->btemp); *temp = ab8500_btemp_get_temp(ab8500_data->btemp);
} else { } else if (sensor == AB8500_SENSOR_BAT_CTRL) {
voltage = ab8500_gpadc_convert(ab8500_data->gpadc, sensor); *temp = ab8500_btemp_get_batctrl_temp(ab8500_data->btemp);
if (voltage < 0) } else if (sensor == AB8500_SENSOR_AUX1) {
return voltage; ret = iio_read_channel_processed(ab8500_data->aux1, &voltage);
if (ret < 0)
return ret;
ret = ab8500_voltage_to_temp(&ab8500_data->cfg, voltage, temp);
if (ret < 0)
return ret;
} else if (sensor == AB8500_SENSOR_AUX2) {
ret = iio_read_channel_processed(ab8500_data->aux2, &voltage);
if (ret < 0)
return ret;
ret = ab8500_voltage_to_temp(&ab8500_data->cfg, voltage, temp); ret = ab8500_voltage_to_temp(&ab8500_data->cfg, voltage, temp);
if (ret < 0) if (ret < 0)
return ret; return ret;
...@@ -164,10 +175,6 @@ int abx500_hwmon_init(struct abx500_temp *data) ...@@ -164,10 +175,6 @@ int abx500_hwmon_init(struct abx500_temp *data)
if (!ab8500_data) if (!ab8500_data)
return -ENOMEM; return -ENOMEM;
ab8500_data->gpadc = ab8500_gpadc_get("ab8500-gpadc.0");
if (IS_ERR(ab8500_data->gpadc))
return PTR_ERR(ab8500_data->gpadc);
ab8500_data->btemp = ab8500_btemp_get(); ab8500_data->btemp = ab8500_btemp_get();
if (IS_ERR(ab8500_data->btemp)) if (IS_ERR(ab8500_data->btemp))
return PTR_ERR(ab8500_data->btemp); return PTR_ERR(ab8500_data->btemp);
...@@ -181,15 +188,25 @@ int abx500_hwmon_init(struct abx500_temp *data) ...@@ -181,15 +188,25 @@ int abx500_hwmon_init(struct abx500_temp *data)
ab8500_data->cfg.tbl_sz = ab8500_temp_tbl_a_size; ab8500_data->cfg.tbl_sz = ab8500_temp_tbl_a_size;
data->plat_data = ab8500_data; data->plat_data = ab8500_data;
ab8500_data->aux1 = devm_iio_channel_get(&data->pdev->dev, "aux1");
if (IS_ERR(ab8500_data->aux1)) {
if (PTR_ERR(ab8500_data->aux1) == -ENODEV)
return -EPROBE_DEFER;
dev_err(&data->pdev->dev, "failed to get AUX1 ADC channel\n");
return PTR_ERR(ab8500_data->aux1);
}
ab8500_data->aux2 = devm_iio_channel_get(&data->pdev->dev, "aux2");
if (IS_ERR(ab8500_data->aux2)) {
if (PTR_ERR(ab8500_data->aux2) == -ENODEV)
return -EPROBE_DEFER;
dev_err(&data->pdev->dev, "failed to get AUX2 ADC channel\n");
return PTR_ERR(ab8500_data->aux2);
}
/* data->gpadc_addr[0] = AB8500_SENSOR_AUX1;
* ADC_AUX1 and ADC_AUX2, connected to external NTC data->gpadc_addr[1] = AB8500_SENSOR_AUX2;
* BTEMP_BALL and BAT_CTRL, fixed usage data->gpadc_addr[2] = AB8500_SENSOR_BTEMP_BALL;
*/ data->gpadc_addr[3] = AB8500_SENSOR_BAT_CTRL;
data->gpadc_addr[0] = ADC_AUX1;
data->gpadc_addr[1] = ADC_AUX2;
data->gpadc_addr[2] = BTEMP_BALL;
data->gpadc_addr[3] = BAT_CTRL;
data->monitored_sensors = NUM_MONITORED_SENSORS; data->monitored_sensors = NUM_MONITORED_SENSORS;
data->ops.read_sensor = ab8500_read_sensor; data->ops.read_sensor = ab8500_read_sensor;
......
...@@ -15,7 +15,6 @@ ...@@ -15,7 +15,6 @@
#include <linux/types.h> #include <linux/types.h>
#include <linux/interrupt.h> #include <linux/interrupt.h>
#include <linux/i2c.h> #include <linux/i2c.h>
#include <linux/gpio.h>
#include <linux/irq.h> #include <linux/irq.h>
#include <linux/iio/iio.h> #include <linux/iio/iio.h>
#include <linux/iio/sysfs.h> #include <linux/iio/sysfs.h>
......
...@@ -6,6 +6,16 @@ ...@@ -6,6 +6,16 @@
menu "Analog to digital converters" menu "Analog to digital converters"
config AB8500_GPADC
bool "ST-Ericsson AB8500 GPADC driver"
depends on AB8500_CORE && REGULATOR_AB8500
default y
help
AB8500 Analog Baseband, mixed signal integrated circuit GPADC
(General Purpose Analog to Digital Converter) driver used to monitor
internal voltages, convert accessory and battery, AC (charger, mains)
and USB voltages integral to the U8500 platform.
config AD_SIGMA_DELTA config AD_SIGMA_DELTA
tristate tristate
select IIO_BUFFER select IIO_BUFFER
...@@ -432,6 +442,17 @@ config INGENIC_ADC ...@@ -432,6 +442,17 @@ config INGENIC_ADC
This driver can also be built as a module. If so, the module will be This driver can also be built as a module. If so, the module will be
called ingenic_adc. called ingenic_adc.
config INTEL_MRFLD_ADC
tristate "Intel Merrifield Basin Cove ADC driver"
depends on INTEL_SOC_PMIC_MRFLD
help
Say yes here to have support for Basin Cove power management IC (PMIC) ADC
device. Depending on platform configuration, this general purpose ADC can
be used for sampling sensors such as thermal resistors.
To compile this driver as a module, choose M here: the module will be
called intel_mrfld_adc.
config IMX7D_ADC config IMX7D_ADC
tristate "Freescale IMX7D ADC driver" tristate "Freescale IMX7D ADC driver"
depends on ARCH_MXC || COMPILE_TEST depends on ARCH_MXC || COMPILE_TEST
...@@ -508,8 +529,8 @@ config MAX1027 ...@@ -508,8 +529,8 @@ config MAX1027
select IIO_BUFFER select IIO_BUFFER
select IIO_TRIGGERED_BUFFER select IIO_TRIGGERED_BUFFER
help help
Say yes here to build support for Maxim SPI ADC models Say yes here to build support for Maxim SPI {10,12}-bit ADC models:
max1027, max1029 and max1031. max1027, max1029, max1031, max1227, max1229 and max1231.
To compile this driver as a module, choose M here: the module will be To compile this driver as a module, choose M here: the module will be
called max1027. called max1027.
......
...@@ -4,6 +4,7 @@ ...@@ -4,6 +4,7 @@
# #
# When adding new entries keep the list in alphabetical order # When adding new entries keep the list in alphabetical order
obj-$(CONFIG_AB8500_GPADC) += ab8500-gpadc.o
obj-$(CONFIG_AD_SIGMA_DELTA) += ad_sigma_delta.o obj-$(CONFIG_AD_SIGMA_DELTA) += ad_sigma_delta.o
obj-$(CONFIG_AD7124) += ad7124.o obj-$(CONFIG_AD7124) += ad7124.o
obj-$(CONFIG_AD7266) += ad7266.o obj-$(CONFIG_AD7266) += ad7266.o
...@@ -42,6 +43,7 @@ obj-$(CONFIG_HX711) += hx711.o ...@@ -42,6 +43,7 @@ obj-$(CONFIG_HX711) += hx711.o
obj-$(CONFIG_IMX7D_ADC) += imx7d_adc.o obj-$(CONFIG_IMX7D_ADC) += imx7d_adc.o
obj-$(CONFIG_INA2XX_ADC) += ina2xx-adc.o obj-$(CONFIG_INA2XX_ADC) += ina2xx-adc.o
obj-$(CONFIG_INGENIC_ADC) += ingenic-adc.o obj-$(CONFIG_INGENIC_ADC) += ingenic-adc.o
obj-$(CONFIG_INTEL_MRFLD_ADC) += intel_mrfld_adc.o
obj-$(CONFIG_LP8788_ADC) += lp8788_adc.o obj-$(CONFIG_LP8788_ADC) += lp8788_adc.o
obj-$(CONFIG_LPC18XX_ADC) += lpc18xx_adc.o obj-$(CONFIG_LPC18XX_ADC) += lpc18xx_adc.o
obj-$(CONFIG_LPC32XX_ADC) += lpc32xx_adc.o obj-$(CONFIG_LPC32XX_ADC) += lpc32xx_adc.o
......
...@@ -1483,7 +1483,7 @@ static void at91_adc_dma_init(struct platform_device *pdev) ...@@ -1483,7 +1483,7 @@ static void at91_adc_dma_init(struct platform_device *pdev)
st->dma_st.rx_buf, st->dma_st.rx_dma_buf); st->dma_st.rx_buf, st->dma_st.rx_dma_buf);
dma_chan_disable: dma_chan_disable:
dma_release_channel(st->dma_st.dma_chan); dma_release_channel(st->dma_st.dma_chan);
st->dma_st.dma_chan = 0; st->dma_st.dma_chan = NULL;
dma_exit: dma_exit:
dev_info(&pdev->dev, "continuing without DMA support\n"); dev_info(&pdev->dev, "continuing without DMA support\n");
} }
...@@ -1506,7 +1506,7 @@ static void at91_adc_dma_disable(struct platform_device *pdev) ...@@ -1506,7 +1506,7 @@ static void at91_adc_dma_disable(struct platform_device *pdev)
dma_free_coherent(st->dma_st.dma_chan->device->dev, pages * PAGE_SIZE, dma_free_coherent(st->dma_st.dma_chan->device->dev, pages * PAGE_SIZE,
st->dma_st.rx_buf, st->dma_st.rx_dma_buf); st->dma_st.rx_buf, st->dma_st.rx_dma_buf);
dma_release_channel(st->dma_st.dma_chan); dma_release_channel(st->dma_st.dma_chan);
st->dma_st.dma_chan = 0; st->dma_st.dma_chan = NULL;
dev_info(&pdev->dev, "continuing without DMA support\n"); dev_info(&pdev->dev, "continuing without DMA support\n");
} }
......
...@@ -1008,7 +1008,7 @@ static int cpcap_adc_probe(struct platform_device *pdev) ...@@ -1008,7 +1008,7 @@ static int cpcap_adc_probe(struct platform_device *pdev)
error = devm_request_threaded_irq(&pdev->dev, ddata->irq, NULL, error = devm_request_threaded_irq(&pdev->dev, ddata->irq, NULL,
cpcap_adc_irq_thread, cpcap_adc_irq_thread,
IRQF_TRIGGER_NONE, IRQF_TRIGGER_NONE | IRQF_ONESHOT,
"cpcap-adc", indio_dev); "cpcap-adc", indio_dev);
if (error) { if (error) {
dev_err(&pdev->dev, "could not get irq: %i\n", dev_err(&pdev->dev, "could not get irq: %i\n",
......
...@@ -651,7 +651,7 @@ static irqreturn_t exynos_ts_isr(int irq, void *dev_id) ...@@ -651,7 +651,7 @@ static irqreturn_t exynos_ts_isr(int irq, void *dev_id)
input_sync(info->input); input_sync(info->input);
usleep_range(1000, 1100); usleep_range(1000, 1100);
}; }
writel(0, ADC_V1_CLRINTPNDNUP(info->regs)); writel(0, ADC_V1_CLRINTPNDNUP(info->regs));
...@@ -769,7 +769,6 @@ static int exynos_adc_probe(struct platform_device *pdev) ...@@ -769,7 +769,6 @@ static int exynos_adc_probe(struct platform_device *pdev)
struct device_node *np = pdev->dev.of_node; struct device_node *np = pdev->dev.of_node;
struct s3c2410_ts_mach_info *pdata = dev_get_platdata(&pdev->dev); struct s3c2410_ts_mach_info *pdata = dev_get_platdata(&pdev->dev);
struct iio_dev *indio_dev = NULL; struct iio_dev *indio_dev = NULL;
struct resource *mem;
bool has_ts = false; bool has_ts = false;
int ret = -ENODEV; int ret = -ENODEV;
int irq; int irq;
...@@ -788,8 +787,7 @@ static int exynos_adc_probe(struct platform_device *pdev) ...@@ -788,8 +787,7 @@ static int exynos_adc_probe(struct platform_device *pdev)
return -EINVAL; return -EINVAL;
} }
mem = platform_get_resource(pdev, IORESOURCE_MEM, 0); info->regs = devm_platform_ioremap_resource(pdev, 0);
info->regs = devm_ioremap_resource(&pdev->dev, mem);
if (IS_ERR(info->regs)) if (IS_ERR(info->regs))
return PTR_ERR(info->regs); return PTR_ERR(info->regs);
......
...@@ -23,6 +23,7 @@ ...@@ -23,6 +23,7 @@
/* gain to pulse and scale conversion */ /* gain to pulse and scale conversion */
#define HX711_GAIN_MAX 3 #define HX711_GAIN_MAX 3
#define HX711_RESET_GAIN 128
struct hx711_gain_to_scale { struct hx711_gain_to_scale {
int gain; int gain;
...@@ -185,8 +186,7 @@ static int hx711_wait_for_ready(struct hx711_data *hx711_data) ...@@ -185,8 +186,7 @@ static int hx711_wait_for_ready(struct hx711_data *hx711_data)
static int hx711_reset(struct hx711_data *hx711_data) static int hx711_reset(struct hx711_data *hx711_data)
{ {
int ret; int val = hx711_wait_for_ready(hx711_data);
int val = gpiod_get_value(hx711_data->gpiod_dout);
if (val) { if (val) {
/* /*
...@@ -202,22 +202,10 @@ static int hx711_reset(struct hx711_data *hx711_data) ...@@ -202,22 +202,10 @@ static int hx711_reset(struct hx711_data *hx711_data)
msleep(10); msleep(10);
gpiod_set_value(hx711_data->gpiod_pd_sck, 0); gpiod_set_value(hx711_data->gpiod_pd_sck, 0);
ret = hx711_wait_for_ready(hx711_data);
if (ret)
return ret;
/*
* after a reset the gain is 128 so we do a dummy read
* to set the gain for the next read
*/
ret = hx711_read(hx711_data);
if (ret < 0)
return ret;
/*
* after a dummy read we need to wait vor readiness
* for not mixing gain pulses with the clock
*/
val = hx711_wait_for_ready(hx711_data); val = hx711_wait_for_ready(hx711_data);
/* after a reset the gain is 128 */
hx711_data->gain_set = HX711_RESET_GAIN;
} }
return val; return val;
......
// SPDX-License-Identifier: GPL-2.0
/*
* ADC driver for Basin Cove PMIC
*
* Copyright (C) 2012 Intel Corporation
* Author: Bin Yang <bin.yang@intel.com>
*
* Rewritten for upstream by:
* Vincent Pelletier <plr.vincent@gmail.com>
* Andy Shevchenko <andriy.shevchenko@linux.intel.com>
*/
#include <linux/bitops.h>
#include <linux/completion.h>
#include <linux/interrupt.h>
#include <linux/mfd/intel_soc_pmic.h>
#include <linux/mfd/intel_soc_pmic_mrfld.h>
#include <linux/module.h>
#include <linux/mutex.h>
#include <linux/platform_device.h>
#include <linux/regmap.h>
#include <linux/iio/driver.h>
#include <linux/iio/iio.h>
#include <linux/iio/machine.h>
#include <asm/unaligned.h>
#define BCOVE_GPADCREQ 0xDC
#define BCOVE_GPADCREQ_BUSY BIT(0)
#define BCOVE_GPADCREQ_IRQEN BIT(1)
#define BCOVE_ADCIRQ_ALL ( \
BCOVE_ADCIRQ_BATTEMP | \
BCOVE_ADCIRQ_SYSTEMP | \
BCOVE_ADCIRQ_BATTID | \
BCOVE_ADCIRQ_VIBATT | \
BCOVE_ADCIRQ_CCTICK)
#define BCOVE_ADC_TIMEOUT msecs_to_jiffies(1000)
static const u8 mrfld_adc_requests[] = {
BCOVE_ADCIRQ_VIBATT,
BCOVE_ADCIRQ_BATTID,
BCOVE_ADCIRQ_VIBATT,
BCOVE_ADCIRQ_SYSTEMP,
BCOVE_ADCIRQ_BATTEMP,
BCOVE_ADCIRQ_BATTEMP,
BCOVE_ADCIRQ_SYSTEMP,
BCOVE_ADCIRQ_SYSTEMP,
BCOVE_ADCIRQ_SYSTEMP,
};
struct mrfld_adc {
struct regmap *regmap;
struct completion completion;
/* Lock to protect the IPC transfers */
struct mutex lock;
};
static irqreturn_t mrfld_adc_thread_isr(int irq, void *data)
{
struct iio_dev *indio_dev = data;
struct mrfld_adc *adc = iio_priv(indio_dev);
complete(&adc->completion);
return IRQ_HANDLED;
}
static int mrfld_adc_single_conv(struct iio_dev *indio_dev,
struct iio_chan_spec const *chan,
int *result)
{
struct mrfld_adc *adc = iio_priv(indio_dev);
struct regmap *regmap = adc->regmap;
unsigned int req;
long timeout;
u8 buf[2];
int ret;
reinit_completion(&adc->completion);
regmap_update_bits(regmap, BCOVE_MADCIRQ, BCOVE_ADCIRQ_ALL, 0);
regmap_update_bits(regmap, BCOVE_MIRQLVL1, BCOVE_LVL1_ADC, 0);
ret = regmap_read_poll_timeout(regmap, BCOVE_GPADCREQ, req,
!(req & BCOVE_GPADCREQ_BUSY),
2000, 1000000);
if (ret)
goto done;
req = mrfld_adc_requests[chan->channel];
ret = regmap_write(regmap, BCOVE_GPADCREQ, BCOVE_GPADCREQ_IRQEN | req);
if (ret)
goto done;
timeout = wait_for_completion_interruptible_timeout(&adc->completion,
BCOVE_ADC_TIMEOUT);
if (timeout < 0) {
ret = timeout;
goto done;
}
if (timeout == 0) {
ret = -ETIMEDOUT;
goto done;
}
ret = regmap_bulk_read(regmap, chan->address, buf, 2);
if (ret)
goto done;
*result = get_unaligned_be16(buf);
ret = IIO_VAL_INT;
done:
regmap_update_bits(regmap, BCOVE_MIRQLVL1, BCOVE_LVL1_ADC, 0xff);
regmap_update_bits(regmap, BCOVE_MADCIRQ, BCOVE_ADCIRQ_ALL, 0xff);
return ret;
}
static int mrfld_adc_read_raw(struct iio_dev *indio_dev,
struct iio_chan_spec const *chan,
int *val, int *val2, long mask)
{
struct mrfld_adc *adc = iio_priv(indio_dev);
int ret;
switch (mask) {
case IIO_CHAN_INFO_RAW:
mutex_lock(&adc->lock);
ret = mrfld_adc_single_conv(indio_dev, chan, val);
mutex_unlock(&adc->lock);
return ret;
default:
return -EINVAL;
}
}
static const struct iio_info mrfld_adc_iio_info = {
.read_raw = &mrfld_adc_read_raw,
};
#define BCOVE_ADC_CHANNEL(_type, _channel, _datasheet_name, _address) \
{ \
.indexed = 1, \
.type = _type, \
.channel = _channel, \
.address = _address, \
.datasheet_name = _datasheet_name, \
.info_mask_separate = BIT(IIO_CHAN_INFO_RAW), \
}
static const struct iio_chan_spec mrfld_adc_channels[] = {
BCOVE_ADC_CHANNEL(IIO_VOLTAGE, 0, "CH0", 0xE9),
BCOVE_ADC_CHANNEL(IIO_RESISTANCE, 1, "CH1", 0xEB),
BCOVE_ADC_CHANNEL(IIO_CURRENT, 2, "CH2", 0xED),
BCOVE_ADC_CHANNEL(IIO_TEMP, 3, "CH3", 0xCC),
BCOVE_ADC_CHANNEL(IIO_TEMP, 4, "CH4", 0xC8),
BCOVE_ADC_CHANNEL(IIO_TEMP, 5, "CH5", 0xCA),
BCOVE_ADC_CHANNEL(IIO_TEMP, 6, "CH6", 0xC2),
BCOVE_ADC_CHANNEL(IIO_TEMP, 7, "CH7", 0xC4),
BCOVE_ADC_CHANNEL(IIO_TEMP, 8, "CH8", 0xC6),
};
static struct iio_map iio_maps[] = {
IIO_MAP("CH0", "bcove-battery", "VBATRSLT"),
IIO_MAP("CH1", "bcove-battery", "BATTID"),
IIO_MAP("CH2", "bcove-battery", "IBATRSLT"),
IIO_MAP("CH3", "bcove-temp", "PMICTEMP"),
IIO_MAP("CH4", "bcove-temp", "BATTEMP0"),
IIO_MAP("CH5", "bcove-temp", "BATTEMP1"),
IIO_MAP("CH6", "bcove-temp", "SYSTEMP0"),
IIO_MAP("CH7", "bcove-temp", "SYSTEMP1"),
IIO_MAP("CH8", "bcove-temp", "SYSTEMP2"),
{}
};
static int mrfld_adc_probe(struct platform_device *pdev)
{
struct device *dev = &pdev->dev;
struct intel_soc_pmic *pmic = dev_get_drvdata(dev->parent);
struct iio_dev *indio_dev;
struct mrfld_adc *adc;
int irq;
int ret;
indio_dev = devm_iio_device_alloc(dev, sizeof(*indio_dev));
if (!indio_dev)
return -ENOMEM;
adc = iio_priv(indio_dev);
mutex_init(&adc->lock);
init_completion(&adc->completion);
adc->regmap = pmic->regmap;
irq = platform_get_irq(pdev, 0);
if (irq < 0)
return irq;
ret = devm_request_threaded_irq(dev, irq, NULL, mrfld_adc_thread_isr,
IRQF_ONESHOT | IRQF_SHARED, pdev->name,
indio_dev);
if (ret)
return ret;
platform_set_drvdata(pdev, indio_dev);
indio_dev->dev.parent = dev;
indio_dev->name = pdev->name;
indio_dev->channels = mrfld_adc_channels;
indio_dev->num_channels = ARRAY_SIZE(mrfld_adc_channels);
indio_dev->info = &mrfld_adc_iio_info;
indio_dev->modes = INDIO_DIRECT_MODE;
ret = iio_map_array_register(indio_dev, iio_maps);
if (ret)
return ret;
ret = devm_iio_device_register(dev, indio_dev);
if (ret < 0)
goto err_array_unregister;
return 0;
err_array_unregister:
iio_map_array_unregister(indio_dev);
return ret;
}
static int mrfld_adc_remove(struct platform_device *pdev)
{
struct iio_dev *indio_dev = platform_get_drvdata(pdev);
iio_map_array_unregister(indio_dev);
return 0;
}
static const struct platform_device_id mrfld_adc_id_table[] = {
{ .name = "mrfld_bcove_adc" },
{}
};
MODULE_DEVICE_TABLE(platform, mrfld_adc_id_table);
static struct platform_driver mrfld_adc_driver = {
.driver = {
.name = "mrfld_bcove_adc",
},
.probe = mrfld_adc_probe,
.remove = mrfld_adc_remove,
.id_table = mrfld_adc_id_table,
};
module_platform_driver(mrfld_adc_driver);
MODULE_AUTHOR("Bin Yang <bin.yang@intel.com>");
MODULE_AUTHOR("Vincent Pelletier <plr.vincent@gmail.com>");
MODULE_AUTHOR("Andy Shevchenko <andriy.shevchenko@linux.intel.com>");
MODULE_DESCRIPTION("ADC driver for Basin Cove PMIC");
MODULE_LICENSE("GPL v2");
...@@ -63,12 +63,18 @@ enum max1027_id { ...@@ -63,12 +63,18 @@ enum max1027_id {
max1027, max1027,
max1029, max1029,
max1031, max1031,
max1227,
max1229,
max1231,
}; };
static const struct spi_device_id max1027_id[] = { static const struct spi_device_id max1027_id[] = {
{"max1027", max1027}, {"max1027", max1027},
{"max1029", max1029}, {"max1029", max1029},
{"max1031", max1031}, {"max1031", max1031},
{"max1227", max1227},
{"max1229", max1229},
{"max1231", max1231},
{} {}
}; };
MODULE_DEVICE_TABLE(spi, max1027_id); MODULE_DEVICE_TABLE(spi, max1027_id);
...@@ -78,12 +84,15 @@ static const struct of_device_id max1027_adc_dt_ids[] = { ...@@ -78,12 +84,15 @@ static const struct of_device_id max1027_adc_dt_ids[] = {
{ .compatible = "maxim,max1027" }, { .compatible = "maxim,max1027" },
{ .compatible = "maxim,max1029" }, { .compatible = "maxim,max1029" },
{ .compatible = "maxim,max1031" }, { .compatible = "maxim,max1031" },
{ .compatible = "maxim,max1227" },
{ .compatible = "maxim,max1229" },
{ .compatible = "maxim,max1231" },
{}, {},
}; };
MODULE_DEVICE_TABLE(of, max1027_adc_dt_ids); MODULE_DEVICE_TABLE(of, max1027_adc_dt_ids);
#endif #endif
#define MAX1027_V_CHAN(index) \ #define MAX1027_V_CHAN(index, depth) \
{ \ { \
.type = IIO_VOLTAGE, \ .type = IIO_VOLTAGE, \
.indexed = 1, \ .indexed = 1, \
...@@ -93,7 +102,7 @@ MODULE_DEVICE_TABLE(of, max1027_adc_dt_ids); ...@@ -93,7 +102,7 @@ MODULE_DEVICE_TABLE(of, max1027_adc_dt_ids);
.scan_index = index + 1, \ .scan_index = index + 1, \
.scan_type = { \ .scan_type = { \
.sign = 'u', \ .sign = 'u', \
.realbits = 10, \ .realbits = depth, \
.storagebits = 16, \ .storagebits = 16, \
.shift = 2, \ .shift = 2, \
.endianness = IIO_BE, \ .endianness = IIO_BE, \
...@@ -115,52 +124,54 @@ MODULE_DEVICE_TABLE(of, max1027_adc_dt_ids); ...@@ -115,52 +124,54 @@ MODULE_DEVICE_TABLE(of, max1027_adc_dt_ids);
}, \ }, \
} }
#define MAX1X27_CHANNELS(depth) \
MAX1027_T_CHAN, \
MAX1027_V_CHAN(0, depth), \
MAX1027_V_CHAN(1, depth), \
MAX1027_V_CHAN(2, depth), \
MAX1027_V_CHAN(3, depth), \
MAX1027_V_CHAN(4, depth), \
MAX1027_V_CHAN(5, depth), \
MAX1027_V_CHAN(6, depth), \
MAX1027_V_CHAN(7, depth)
#define MAX1X29_CHANNELS(depth) \
MAX1X27_CHANNELS(depth), \
MAX1027_V_CHAN(8, depth), \
MAX1027_V_CHAN(9, depth), \
MAX1027_V_CHAN(10, depth), \
MAX1027_V_CHAN(11, depth)
#define MAX1X31_CHANNELS(depth) \
MAX1X27_CHANNELS(depth), \
MAX1X29_CHANNELS(depth), \
MAX1027_V_CHAN(12, depth), \
MAX1027_V_CHAN(13, depth), \
MAX1027_V_CHAN(14, depth), \
MAX1027_V_CHAN(15, depth)
static const struct iio_chan_spec max1027_channels[] = { static const struct iio_chan_spec max1027_channels[] = {
MAX1027_T_CHAN, MAX1X27_CHANNELS(10),
MAX1027_V_CHAN(0),
MAX1027_V_CHAN(1),
MAX1027_V_CHAN(2),
MAX1027_V_CHAN(3),
MAX1027_V_CHAN(4),
MAX1027_V_CHAN(5),
MAX1027_V_CHAN(6),
MAX1027_V_CHAN(7)
}; };
static const struct iio_chan_spec max1029_channels[] = { static const struct iio_chan_spec max1029_channels[] = {
MAX1027_T_CHAN, MAX1X29_CHANNELS(10),
MAX1027_V_CHAN(0),
MAX1027_V_CHAN(1),
MAX1027_V_CHAN(2),
MAX1027_V_CHAN(3),
MAX1027_V_CHAN(4),
MAX1027_V_CHAN(5),
MAX1027_V_CHAN(6),
MAX1027_V_CHAN(7),
MAX1027_V_CHAN(8),
MAX1027_V_CHAN(9),
MAX1027_V_CHAN(10),
MAX1027_V_CHAN(11)
}; };
static const struct iio_chan_spec max1031_channels[] = { static const struct iio_chan_spec max1031_channels[] = {
MAX1027_T_CHAN, MAX1X31_CHANNELS(10),
MAX1027_V_CHAN(0), };
MAX1027_V_CHAN(1),
MAX1027_V_CHAN(2), static const struct iio_chan_spec max1227_channels[] = {
MAX1027_V_CHAN(3), MAX1X27_CHANNELS(12),
MAX1027_V_CHAN(4), };
MAX1027_V_CHAN(5),
MAX1027_V_CHAN(6), static const struct iio_chan_spec max1229_channels[] = {
MAX1027_V_CHAN(7), MAX1X29_CHANNELS(12),
MAX1027_V_CHAN(8), };
MAX1027_V_CHAN(9),
MAX1027_V_CHAN(10), static const struct iio_chan_spec max1231_channels[] = {
MAX1027_V_CHAN(11), MAX1X31_CHANNELS(12),
MAX1027_V_CHAN(12),
MAX1027_V_CHAN(13),
MAX1027_V_CHAN(14),
MAX1027_V_CHAN(15)
}; };
static const unsigned long max1027_available_scan_masks[] = { static const unsigned long max1027_available_scan_masks[] = {
...@@ -200,6 +211,21 @@ static const struct max1027_chip_info max1027_chip_info_tbl[] = { ...@@ -200,6 +211,21 @@ static const struct max1027_chip_info max1027_chip_info_tbl[] = {
.num_channels = ARRAY_SIZE(max1031_channels), .num_channels = ARRAY_SIZE(max1031_channels),
.available_scan_masks = max1031_available_scan_masks, .available_scan_masks = max1031_available_scan_masks,
}, },
[max1227] = {
.channels = max1227_channels,
.num_channels = ARRAY_SIZE(max1227_channels),
.available_scan_masks = max1027_available_scan_masks,
},
[max1229] = {
.channels = max1229_channels,
.num_channels = ARRAY_SIZE(max1229_channels),
.available_scan_masks = max1029_available_scan_masks,
},
[max1231] = {
.channels = max1231_channels,
.num_channels = ARRAY_SIZE(max1231_channels),
.available_scan_masks = max1031_available_scan_masks,
},
}; };
struct max1027_state { struct max1027_state {
...@@ -284,7 +310,7 @@ static int max1027_read_raw(struct iio_dev *indio_dev, ...@@ -284,7 +310,7 @@ static int max1027_read_raw(struct iio_dev *indio_dev,
break; break;
case IIO_VOLTAGE: case IIO_VOLTAGE:
*val = 2500; *val = 2500;
*val2 = 10; *val2 = chan->scan_type.realbits;
ret = IIO_VAL_FRACTIONAL_LOG2; ret = IIO_VAL_FRACTIONAL_LOG2;
break; break;
default: default:
...@@ -309,8 +335,11 @@ static int max1027_debugfs_reg_access(struct iio_dev *indio_dev, ...@@ -309,8 +335,11 @@ static int max1027_debugfs_reg_access(struct iio_dev *indio_dev,
struct max1027_state *st = iio_priv(indio_dev); struct max1027_state *st = iio_priv(indio_dev);
u8 *val = (u8 *)st->buffer; u8 *val = (u8 *)st->buffer;
if (readval != NULL) if (readval) {
return -EINVAL; int ret = spi_read(st->spi, val, 2);
*readval = be16_to_cpu(st->buffer[0]);
return ret;
}
*val = (u8)writeval; *val = (u8)writeval;
return spi_write(st->spi, val, 1); return spi_write(st->spi, val, 1);
...@@ -427,9 +456,11 @@ static int max1027_probe(struct spi_device *spi) ...@@ -427,9 +456,11 @@ static int max1027_probe(struct spi_device *spi)
return -ENOMEM; return -ENOMEM;
} }
if (spi->irq) {
ret = devm_iio_triggered_buffer_setup(&spi->dev, indio_dev, ret = devm_iio_triggered_buffer_setup(&spi->dev, indio_dev,
&iio_pollfunc_store_time, &iio_pollfunc_store_time,
&max1027_trigger_handler, NULL); &max1027_trigger_handler,
NULL);
if (ret < 0) { if (ret < 0) {
dev_err(&indio_dev->dev, "Failed to setup buffer\n"); dev_err(&indio_dev->dev, "Failed to setup buffer\n");
return ret; return ret;
...@@ -439,7 +470,8 @@ static int max1027_probe(struct spi_device *spi) ...@@ -439,7 +470,8 @@ static int max1027_probe(struct spi_device *spi)
indio_dev->name); indio_dev->name);
if (st->trig == NULL) { if (st->trig == NULL) {
ret = -ENOMEM; ret = -ENOMEM;
dev_err(&indio_dev->dev, "Failed to allocate iio trigger\n"); dev_err(&indio_dev->dev,
"Failed to allocate iio trigger\n");
return ret; return ret;
} }
...@@ -452,11 +484,21 @@ static int max1027_probe(struct spi_device *spi) ...@@ -452,11 +484,21 @@ static int max1027_probe(struct spi_device *spi)
iio_trigger_generic_data_rdy_poll, iio_trigger_generic_data_rdy_poll,
NULL, NULL,
IRQF_TRIGGER_FALLING, IRQF_TRIGGER_FALLING,
spi->dev.driver->name, st->trig); spi->dev.driver->name,
st->trig);
if (ret < 0) { if (ret < 0) {
dev_err(&indio_dev->dev, "Failed to allocate IRQ.\n"); dev_err(&indio_dev->dev, "Failed to allocate IRQ.\n");
return ret; return ret;
} }
}
/* Internal reset */
st->reg = MAX1027_RST_REG;
ret = spi_write(st->spi, &st->reg, 1);
if (ret < 0) {
dev_err(&indio_dev->dev, "Failed to reset the ADC\n");
return ret;
}
/* Disable averaging */ /* Disable averaging */
st->reg = MAX1027_AVG_REG; st->reg = MAX1027_AVG_REG;
...@@ -480,5 +522,5 @@ static struct spi_driver max1027_driver = { ...@@ -480,5 +522,5 @@ static struct spi_driver max1027_driver = {
module_spi_driver(max1027_driver); module_spi_driver(max1027_driver);
MODULE_AUTHOR("Philippe Reynes <tremyfr@yahoo.fr>"); MODULE_AUTHOR("Philippe Reynes <tremyfr@yahoo.fr>");
MODULE_DESCRIPTION("MAX1027/MAX1029/MAX1031 ADC"); MODULE_DESCRIPTION("MAX1X27/MAX1X29/MAX1X31 ADC");
MODULE_LICENSE("GPL v2"); MODULE_LICENSE("GPL v2");
...@@ -164,7 +164,7 @@ static int mcp320x_adc_conversion(struct mcp320x *adc, u8 channel, ...@@ -164,7 +164,7 @@ static int mcp320x_adc_conversion(struct mcp320x *adc, u8 channel,
case mcp3550_60: case mcp3550_60:
case mcp3551: case mcp3551:
case mcp3553: { case mcp3553: {
u32 raw = be32_to_cpup((u32 *)adc->rx_buf); u32 raw = be32_to_cpup((__be32 *)adc->rx_buf);
if (!(adc->spi->mode & SPI_CPOL)) if (!(adc->spi->mode & SPI_CPOL))
raw <<= 1; /* strip Data Ready bit in SPI mode 0,0 */ raw <<= 1; /* strip Data Ready bit in SPI mode 0,0 */
......
...@@ -1187,7 +1187,6 @@ static int meson_sar_adc_probe(struct platform_device *pdev) ...@@ -1187,7 +1187,6 @@ static int meson_sar_adc_probe(struct platform_device *pdev)
const struct meson_sar_adc_data *match_data; const struct meson_sar_adc_data *match_data;
struct meson_sar_adc_priv *priv; struct meson_sar_adc_priv *priv;
struct iio_dev *indio_dev; struct iio_dev *indio_dev;
struct resource *res;
void __iomem *base; void __iomem *base;
int irq, ret; int irq, ret;
...@@ -1214,8 +1213,7 @@ static int meson_sar_adc_probe(struct platform_device *pdev) ...@@ -1214,8 +1213,7 @@ static int meson_sar_adc_probe(struct platform_device *pdev)
indio_dev->modes = INDIO_DIRECT_MODE; indio_dev->modes = INDIO_DIRECT_MODE;
indio_dev->info = &meson_sar_adc_iio_info; indio_dev->info = &meson_sar_adc_iio_info;
res = platform_get_resource(pdev, IORESOURCE_MEM, 0); base = devm_platform_ioremap_resource(pdev, 0);
base = devm_ioremap_resource(&pdev->dev, res);
if (IS_ERR(base)) if (IS_ERR(base))
return PTR_ERR(base); return PTR_ERR(base);
......
...@@ -38,12 +38,12 @@ ...@@ -38,12 +38,12 @@
#define HAS_ANASWVDD BIT(1) #define HAS_ANASWVDD BIT(1)
/** /**
* stm32_adc_common_regs - stm32 common registers, compatible dependent data * struct stm32_adc_common_regs - stm32 common registers
* @csr: common status register offset * @csr: common status register offset
* @ccr: common control register offset * @ccr: common control register offset
* @eoc1: adc1 end of conversion flag in @csr * @eoc1_msk: adc1 end of conversion flag in @csr
* @eoc2: adc2 end of conversion flag in @csr * @eoc2_msk: adc2 end of conversion flag in @csr
* @eoc3: adc3 end of conversion flag in @csr * @eoc3_msk: adc3 end of conversion flag in @csr
* @ier: interrupt enable register offset for each adc * @ier: interrupt enable register offset for each adc
* @eocie_msk: end of conversion interrupt enable mask in @ier * @eocie_msk: end of conversion interrupt enable mask in @ier
*/ */
...@@ -60,7 +60,7 @@ struct stm32_adc_common_regs { ...@@ -60,7 +60,7 @@ struct stm32_adc_common_regs {
struct stm32_adc_priv; struct stm32_adc_priv;
/** /**
* stm32_adc_priv_cfg - stm32 core compatible configuration data * struct stm32_adc_priv_cfg - stm32 core compatible configuration data
* @regs: common registers for all instances * @regs: common registers for all instances
* @clk_sel: clock selection routine * @clk_sel: clock selection routine
* @max_clk_rate_hz: maximum analog clock rate (Hz, from datasheet) * @max_clk_rate_hz: maximum analog clock rate (Hz, from datasheet)
...@@ -117,6 +117,7 @@ static int stm32f4_pclk_div[] = {2, 4, 6, 8}; ...@@ -117,6 +117,7 @@ static int stm32f4_pclk_div[] = {2, 4, 6, 8};
/** /**
* stm32f4_adc_clk_sel() - Select stm32f4 ADC common clock prescaler * stm32f4_adc_clk_sel() - Select stm32f4 ADC common clock prescaler
* @pdev: platform device
* @priv: stm32 ADC core private data * @priv: stm32 ADC core private data
* Select clock prescaler used for analog conversions, before using ADC. * Select clock prescaler used for analog conversions, before using ADC.
*/ */
......
...@@ -102,7 +102,7 @@ struct stm32_adc_calib { ...@@ -102,7 +102,7 @@ struct stm32_adc_calib {
}; };
/** /**
* stm32_adc_regs - stm32 ADC misc registers & bitfield desc * struct stm32_adc_regs - stm32 ADC misc registers & bitfield desc
* @reg: register offset * @reg: register offset
* @mask: bitfield mask * @mask: bitfield mask
* @shift: left shift * @shift: left shift
...@@ -114,7 +114,7 @@ struct stm32_adc_regs { ...@@ -114,7 +114,7 @@ struct stm32_adc_regs {
}; };
/** /**
* stm32_adc_regspec - stm32 registers definition, compatible dependent data * struct stm32_adc_regspec - stm32 registers definition
* @dr: data register offset * @dr: data register offset
* @ier_eoc: interrupt enable register & eocie bitfield * @ier_eoc: interrupt enable register & eocie bitfield
* @isr_eoc: interrupt status register & eoc bitfield * @isr_eoc: interrupt status register & eoc bitfield
...@@ -140,7 +140,7 @@ struct stm32_adc_regspec { ...@@ -140,7 +140,7 @@ struct stm32_adc_regspec {
struct stm32_adc; struct stm32_adc;
/** /**
* stm32_adc_cfg - stm32 compatible configuration data * struct stm32_adc_cfg - stm32 compatible configuration data
* @regs: registers descriptions * @regs: registers descriptions
* @adc_info: per instance input channels definitions * @adc_info: per instance input channels definitions
* @trigs: external trigger sources * @trigs: external trigger sources
...@@ -183,8 +183,8 @@ struct stm32_adc_cfg { ...@@ -183,8 +183,8 @@ struct stm32_adc_cfg {
* @rx_buf: dma rx buffer cpu address * @rx_buf: dma rx buffer cpu address
* @rx_dma_buf: dma rx buffer bus address * @rx_dma_buf: dma rx buffer bus address
* @rx_buf_sz: dma rx buffer size * @rx_buf_sz: dma rx buffer size
* @difsel bitmask to set single-ended/differential channel * @difsel: bitmask to set single-ended/differential channel
* @pcsel bitmask to preselect channels on some devices * @pcsel: bitmask to preselect channels on some devices
* @smpr_val: sampling time settings (e.g. smpr1 / smpr2) * @smpr_val: sampling time settings (e.g. smpr1 / smpr2)
* @cal: optional calibration data on some devices * @cal: optional calibration data on some devices
* @chan_name: channel name array * @chan_name: channel name array
...@@ -254,7 +254,7 @@ static const struct stm32_adc_info stm32h7_adc_info = { ...@@ -254,7 +254,7 @@ static const struct stm32_adc_info stm32h7_adc_info = {
.num_res = ARRAY_SIZE(stm32h7_adc_resolutions), .num_res = ARRAY_SIZE(stm32h7_adc_resolutions),
}; };
/** /*
* stm32f4_sq - describe regular sequence registers * stm32f4_sq - describe regular sequence registers
* - L: sequence len (register & bit field) * - L: sequence len (register & bit field)
* - SQ1..SQ16: sequence entries (register & bit field) * - SQ1..SQ16: sequence entries (register & bit field)
...@@ -301,7 +301,7 @@ static struct stm32_adc_trig_info stm32f4_adc_trigs[] = { ...@@ -301,7 +301,7 @@ static struct stm32_adc_trig_info stm32f4_adc_trigs[] = {
{}, /* sentinel */ {}, /* sentinel */
}; };
/** /*
* stm32f4_smp_bits[] - describe sampling time register index & bit fields * stm32f4_smp_bits[] - describe sampling time register index & bit fields
* Sorted so it can be indexed by channel number. * Sorted so it can be indexed by channel number.
*/ */
...@@ -392,7 +392,7 @@ static struct stm32_adc_trig_info stm32h7_adc_trigs[] = { ...@@ -392,7 +392,7 @@ static struct stm32_adc_trig_info stm32h7_adc_trigs[] = {
{}, {},
}; };
/** /*
* stm32h7_smp_bits - describe sampling time register index & bit fields * stm32h7_smp_bits - describe sampling time register index & bit fields
* Sorted so it can be indexed by channel number. * Sorted so it can be indexed by channel number.
*/ */
...@@ -994,6 +994,7 @@ static int stm32_adc_conf_scan_seq(struct iio_dev *indio_dev, ...@@ -994,6 +994,7 @@ static int stm32_adc_conf_scan_seq(struct iio_dev *indio_dev,
/** /**
* stm32_adc_get_trig_extsel() - Get external trigger selection * stm32_adc_get_trig_extsel() - Get external trigger selection
* @indio_dev: IIO device structure
* @trig: trigger * @trig: trigger
* *
* Returns trigger extsel value, if trig matches, -EINVAL otherwise. * Returns trigger extsel value, if trig matches, -EINVAL otherwise.
...@@ -1297,6 +1298,10 @@ static int stm32_adc_of_xlate(struct iio_dev *indio_dev, ...@@ -1297,6 +1298,10 @@ static int stm32_adc_of_xlate(struct iio_dev *indio_dev,
/** /**
* stm32_adc_debugfs_reg_access - read or write register value * stm32_adc_debugfs_reg_access - read or write register value
* @indio_dev: IIO device structure
* @reg: register offset
* @writeval: value to write
* @readval: value to read
* *
* To read a value from an ADC register: * To read a value from an ADC register:
* echo [ADC reg offset] > direct_reg_access * echo [ADC reg offset] > direct_reg_access
......
...@@ -175,7 +175,7 @@ static int stmpe_read_raw(struct iio_dev *indio_dev, ...@@ -175,7 +175,7 @@ static int stmpe_read_raw(struct iio_dev *indio_dev,
static irqreturn_t stmpe_adc_isr(int irq, void *dev_id) static irqreturn_t stmpe_adc_isr(int irq, void *dev_id)
{ {
struct stmpe_adc *info = (struct stmpe_adc *)dev_id; struct stmpe_adc *info = (struct stmpe_adc *)dev_id;
u16 data; __be16 data;
if (info->channel <= STMPE_ADC_LAST_NR) { if (info->channel <= STMPE_ADC_LAST_NR) {
int int_sta; int int_sta;
......
...@@ -495,7 +495,7 @@ static irqreturn_t twl4030_madc_threaded_irq_handler(int irq, void *_madc) ...@@ -495,7 +495,7 @@ static irqreturn_t twl4030_madc_threaded_irq_handler(int irq, void *_madc)
ret = twl4030_madc_disable_irq(madc, i); ret = twl4030_madc_disable_irq(madc, i);
if (ret < 0) if (ret < 0)
dev_dbg(madc->dev, "Disable interrupt failed %d\n", i); dev_dbg(madc->dev, "Disable interrupt failed %d\n", i);
madc->requests[i].result_pending = 1; madc->requests[i].result_pending = true;
} }
for (i = 0; i < TWL4030_MADC_NUM_METHODS; i++) { for (i = 0; i < TWL4030_MADC_NUM_METHODS; i++) {
r = &madc->requests[i]; r = &madc->requests[i];
...@@ -507,8 +507,8 @@ static irqreturn_t twl4030_madc_threaded_irq_handler(int irq, void *_madc) ...@@ -507,8 +507,8 @@ static irqreturn_t twl4030_madc_threaded_irq_handler(int irq, void *_madc)
len = twl4030_madc_read_channels(madc, method->rbase, len = twl4030_madc_read_channels(madc, method->rbase,
r->channels, r->rbuf, r->raw); r->channels, r->rbuf, r->raw);
/* Free request */ /* Free request */
r->result_pending = 0; r->result_pending = false;
r->active = 0; r->active = false;
} }
mutex_unlock(&madc->lock); mutex_unlock(&madc->lock);
...@@ -521,15 +521,15 @@ static irqreturn_t twl4030_madc_threaded_irq_handler(int irq, void *_madc) ...@@ -521,15 +521,15 @@ static irqreturn_t twl4030_madc_threaded_irq_handler(int irq, void *_madc)
*/ */
for (i = 0; i < TWL4030_MADC_NUM_METHODS; i++) { for (i = 0; i < TWL4030_MADC_NUM_METHODS; i++) {
r = &madc->requests[i]; r = &madc->requests[i];
if (r->active == 0) if (!r->active)
continue; continue;
method = &twl4030_conversion_methods[r->method]; method = &twl4030_conversion_methods[r->method];
/* Read results */ /* Read results */
len = twl4030_madc_read_channels(madc, method->rbase, len = twl4030_madc_read_channels(madc, method->rbase,
r->channels, r->rbuf, r->raw); r->channels, r->rbuf, r->raw);
/* Free request */ /* Free request */
r->result_pending = 0; r->result_pending = false;
r->active = 0; r->active = false;
} }
mutex_unlock(&madc->lock); mutex_unlock(&madc->lock);
...@@ -652,16 +652,16 @@ static int twl4030_madc_conversion(struct twl4030_madc_request *req) ...@@ -652,16 +652,16 @@ static int twl4030_madc_conversion(struct twl4030_madc_request *req)
ret = twl4030_madc_start_conversion(twl4030_madc, req->method); ret = twl4030_madc_start_conversion(twl4030_madc, req->method);
if (ret < 0) if (ret < 0)
goto out; goto out;
twl4030_madc->requests[req->method].active = 1; twl4030_madc->requests[req->method].active = true;
/* Wait until conversion is ready (ctrl register returns EOC) */ /* Wait until conversion is ready (ctrl register returns EOC) */
ret = twl4030_madc_wait_conversion_ready(twl4030_madc, 5, method->ctrl); ret = twl4030_madc_wait_conversion_ready(twl4030_madc, 5, method->ctrl);
if (ret) { if (ret) {
twl4030_madc->requests[req->method].active = 0; twl4030_madc->requests[req->method].active = false;
goto out; goto out;
} }
ret = twl4030_madc_read_channels(twl4030_madc, method->rbase, ret = twl4030_madc_read_channels(twl4030_madc, method->rbase,
req->channels, req->rbuf, req->raw); req->channels, req->rbuf, req->raw);
twl4030_madc->requests[req->method].active = 0; twl4030_madc->requests[req->method].active = false;
out: out:
mutex_unlock(&twl4030_madc->lock); mutex_unlock(&twl4030_madc->lock);
......
...@@ -1150,7 +1150,6 @@ static int xadc_probe(struct platform_device *pdev) ...@@ -1150,7 +1150,6 @@ static int xadc_probe(struct platform_device *pdev)
const struct of_device_id *id; const struct of_device_id *id;
struct iio_dev *indio_dev; struct iio_dev *indio_dev;
unsigned int bipolar_mask; unsigned int bipolar_mask;
struct resource *mem;
unsigned int conf0; unsigned int conf0;
struct xadc *xadc; struct xadc *xadc;
int ret; int ret;
...@@ -1180,8 +1179,7 @@ static int xadc_probe(struct platform_device *pdev) ...@@ -1180,8 +1179,7 @@ static int xadc_probe(struct platform_device *pdev)
spin_lock_init(&xadc->lock); spin_lock_init(&xadc->lock);
INIT_DELAYED_WORK(&xadc->zynq_unmask_work, xadc_zynq_unmask_worker); INIT_DELAYED_WORK(&xadc->zynq_unmask_work, xadc_zynq_unmask_worker);
mem = platform_get_resource(pdev, IORESOURCE_MEM, 0); xadc->base = devm_platform_ioremap_resource(pdev, 0);
xadc->base = devm_ioremap_resource(&pdev->dev, mem);
if (IS_ERR(xadc->base)) if (IS_ERR(xadc->base))
return PTR_ERR(xadc->base); return PTR_ERR(xadc->base);
......
...@@ -483,7 +483,7 @@ static void sgp_init(struct sgp_data *data) ...@@ -483,7 +483,7 @@ static void sgp_init(struct sgp_data *data)
data->iaq_defval_skip_jiffies = data->iaq_defval_skip_jiffies =
43 * data->measure_interval_jiffies; 43 * data->measure_interval_jiffies;
break; break;
}; }
} }
static const struct iio_info sgp_info = { static const struct iio_info sgp_info = {
......
...@@ -117,7 +117,7 @@ static int sps30_do_cmd(struct sps30_state *state, u16 cmd, u8 *data, int size) ...@@ -117,7 +117,7 @@ static int sps30_do_cmd(struct sps30_state *state, u16 cmd, u8 *data, int size)
break; break;
case SPS30_READ_AUTO_CLEANING_PERIOD: case SPS30_READ_AUTO_CLEANING_PERIOD:
buf[0] = SPS30_AUTO_CLEANING_PERIOD >> 8; buf[0] = SPS30_AUTO_CLEANING_PERIOD >> 8;
buf[1] = (u8)SPS30_AUTO_CLEANING_PERIOD; buf[1] = (u8)(SPS30_AUTO_CLEANING_PERIOD & 0xff);
/* fall through */ /* fall through */
case SPS30_READ_DATA_READY_FLAG: case SPS30_READ_DATA_READY_FLAG:
case SPS30_READ_DATA: case SPS30_READ_DATA:
......
...@@ -41,6 +41,7 @@ struct ad7303_state { ...@@ -41,6 +41,7 @@ struct ad7303_state {
struct regulator *vdd_reg; struct regulator *vdd_reg;
struct regulator *vref_reg; struct regulator *vref_reg;
struct mutex lock;
/* /*
* DMA (thus cache coherency maintenance) requires the * DMA (thus cache coherency maintenance) requires the
* transfer buffers to live in their own cache lines. * transfer buffers to live in their own cache lines.
...@@ -79,7 +80,7 @@ static ssize_t ad7303_write_dac_powerdown(struct iio_dev *indio_dev, ...@@ -79,7 +80,7 @@ static ssize_t ad7303_write_dac_powerdown(struct iio_dev *indio_dev,
if (ret) if (ret)
return ret; return ret;
mutex_lock(&indio_dev->mlock); mutex_lock(&st->lock);
if (pwr_down) if (pwr_down)
st->config |= AD7303_CFG_POWER_DOWN(chan->channel); st->config |= AD7303_CFG_POWER_DOWN(chan->channel);
...@@ -90,7 +91,7 @@ static ssize_t ad7303_write_dac_powerdown(struct iio_dev *indio_dev, ...@@ -90,7 +91,7 @@ static ssize_t ad7303_write_dac_powerdown(struct iio_dev *indio_dev,
* mode, so just write one of the DAC channels again */ * mode, so just write one of the DAC channels again */
ad7303_write(st, chan->channel, st->dac_cache[chan->channel]); ad7303_write(st, chan->channel, st->dac_cache[chan->channel]);
mutex_unlock(&indio_dev->mlock); mutex_unlock(&st->lock);
return len; return len;
} }
...@@ -116,7 +117,9 @@ static int ad7303_read_raw(struct iio_dev *indio_dev, ...@@ -116,7 +117,9 @@ static int ad7303_read_raw(struct iio_dev *indio_dev,
switch (info) { switch (info) {
case IIO_CHAN_INFO_RAW: case IIO_CHAN_INFO_RAW:
mutex_lock(&st->lock);
*val = st->dac_cache[chan->channel]; *val = st->dac_cache[chan->channel];
mutex_unlock(&st->lock);
return IIO_VAL_INT; return IIO_VAL_INT;
case IIO_CHAN_INFO_SCALE: case IIO_CHAN_INFO_SCALE:
vref_uv = ad7303_get_vref(st, chan); vref_uv = ad7303_get_vref(st, chan);
...@@ -144,11 +147,11 @@ static int ad7303_write_raw(struct iio_dev *indio_dev, ...@@ -144,11 +147,11 @@ static int ad7303_write_raw(struct iio_dev *indio_dev,
if (val >= (1 << chan->scan_type.realbits) || val < 0) if (val >= (1 << chan->scan_type.realbits) || val < 0)
return -EINVAL; return -EINVAL;
mutex_lock(&indio_dev->mlock); mutex_lock(&st->lock);
ret = ad7303_write(st, chan->address, val); ret = ad7303_write(st, chan->address, val);
if (ret == 0) if (ret == 0)
st->dac_cache[chan->channel] = val; st->dac_cache[chan->channel] = val;
mutex_unlock(&indio_dev->mlock); mutex_unlock(&st->lock);
break; break;
default: default:
ret = -EINVAL; ret = -EINVAL;
...@@ -211,6 +214,8 @@ static int ad7303_probe(struct spi_device *spi) ...@@ -211,6 +214,8 @@ static int ad7303_probe(struct spi_device *spi)
st->spi = spi; st->spi = spi;
mutex_init(&st->lock);
st->vdd_reg = devm_regulator_get(&spi->dev, "Vdd"); st->vdd_reg = devm_regulator_get(&spi->dev, "Vdd");
if (IS_ERR(st->vdd_reg)) if (IS_ERR(st->vdd_reg))
return PTR_ERR(st->vdd_reg); return PTR_ERR(st->vdd_reg);
......
...@@ -11,6 +11,7 @@ ...@@ -11,6 +11,7 @@
#include <linux/delay.h> #include <linux/delay.h>
#include <linux/module.h> #include <linux/module.h>
#include <linux/of_platform.h> #include <linux/of_platform.h>
#include <linux/pm_runtime.h>
#include <linux/regulator/consumer.h> #include <linux/regulator/consumer.h>
#include <linux/reset.h> #include <linux/reset.h>
...@@ -50,6 +51,41 @@ static const struct regmap_config stm32_dac_regmap_cfg = { ...@@ -50,6 +51,41 @@ static const struct regmap_config stm32_dac_regmap_cfg = {
.max_register = 0x3fc, .max_register = 0x3fc,
}; };
static int stm32_dac_core_hw_start(struct device *dev)
{
struct stm32_dac_common *common = dev_get_drvdata(dev);
struct stm32_dac_priv *priv = to_stm32_dac_priv(common);
int ret;
ret = regulator_enable(priv->vref);
if (ret < 0) {
dev_err(dev, "vref enable failed: %d\n", ret);
return ret;
}
ret = clk_prepare_enable(priv->pclk);
if (ret < 0) {
dev_err(dev, "pclk enable failed: %d\n", ret);
goto err_regulator_disable;
}
return 0;
err_regulator_disable:
regulator_disable(priv->vref);
return ret;
}
static void stm32_dac_core_hw_stop(struct device *dev)
{
struct stm32_dac_common *common = dev_get_drvdata(dev);
struct stm32_dac_priv *priv = to_stm32_dac_priv(common);
clk_disable_unprepare(priv->pclk);
regulator_disable(priv->vref);
}
static int stm32_dac_probe(struct platform_device *pdev) static int stm32_dac_probe(struct platform_device *pdev)
{ {
struct device *dev = &pdev->dev; struct device *dev = &pdev->dev;
...@@ -66,6 +102,8 @@ static int stm32_dac_probe(struct platform_device *pdev) ...@@ -66,6 +102,8 @@ static int stm32_dac_probe(struct platform_device *pdev)
priv = devm_kzalloc(dev, sizeof(*priv), GFP_KERNEL); priv = devm_kzalloc(dev, sizeof(*priv), GFP_KERNEL);
if (!priv) if (!priv)
return -ENOMEM; return -ENOMEM;
platform_set_drvdata(pdev, &priv->common);
cfg = (const struct stm32_dac_cfg *) cfg = (const struct stm32_dac_cfg *)
of_match_device(dev->driver->of_match_table, dev)->data; of_match_device(dev->driver->of_match_table, dev)->data;
...@@ -74,11 +112,19 @@ static int stm32_dac_probe(struct platform_device *pdev) ...@@ -74,11 +112,19 @@ static int stm32_dac_probe(struct platform_device *pdev)
if (IS_ERR(mmio)) if (IS_ERR(mmio))
return PTR_ERR(mmio); return PTR_ERR(mmio);
regmap = devm_regmap_init_mmio(dev, mmio, &stm32_dac_regmap_cfg); regmap = devm_regmap_init_mmio_clk(dev, "pclk", mmio,
&stm32_dac_regmap_cfg);
if (IS_ERR(regmap)) if (IS_ERR(regmap))
return PTR_ERR(regmap); return PTR_ERR(regmap);
priv->common.regmap = regmap; priv->common.regmap = regmap;
priv->pclk = devm_clk_get(dev, "pclk");
if (IS_ERR(priv->pclk)) {
ret = PTR_ERR(priv->pclk);
dev_err(dev, "pclk get failed\n");
return ret;
}
priv->vref = devm_regulator_get(dev, "vref"); priv->vref = devm_regulator_get(dev, "vref");
if (IS_ERR(priv->vref)) { if (IS_ERR(priv->vref)) {
ret = PTR_ERR(priv->vref); ret = PTR_ERR(priv->vref);
...@@ -86,33 +132,22 @@ static int stm32_dac_probe(struct platform_device *pdev) ...@@ -86,33 +132,22 @@ static int stm32_dac_probe(struct platform_device *pdev)
return ret; return ret;
} }
ret = regulator_enable(priv->vref); pm_runtime_get_noresume(dev);
if (ret < 0) { pm_runtime_set_active(dev);
dev_err(dev, "vref enable failed\n"); pm_runtime_enable(dev);
return ret;
} ret = stm32_dac_core_hw_start(dev);
if (ret)
goto err_pm_stop;
ret = regulator_get_voltage(priv->vref); ret = regulator_get_voltage(priv->vref);
if (ret < 0) { if (ret < 0) {
dev_err(dev, "vref get voltage failed, %d\n", ret); dev_err(dev, "vref get voltage failed, %d\n", ret);
goto err_vref; goto err_hw_stop;
} }
priv->common.vref_mv = ret / 1000; priv->common.vref_mv = ret / 1000;
dev_dbg(dev, "vref+=%dmV\n", priv->common.vref_mv); dev_dbg(dev, "vref+=%dmV\n", priv->common.vref_mv);
priv->pclk = devm_clk_get(dev, "pclk");
if (IS_ERR(priv->pclk)) {
ret = PTR_ERR(priv->pclk);
dev_err(dev, "pclk get failed\n");
goto err_vref;
}
ret = clk_prepare_enable(priv->pclk);
if (ret < 0) {
dev_err(dev, "pclk enable failed\n");
goto err_vref;
}
priv->rst = devm_reset_control_get_exclusive(dev, NULL); priv->rst = devm_reset_control_get_exclusive(dev, NULL);
if (!IS_ERR(priv->rst)) { if (!IS_ERR(priv->rst)) {
reset_control_assert(priv->rst); reset_control_assert(priv->rst);
...@@ -128,39 +163,79 @@ static int stm32_dac_probe(struct platform_device *pdev) ...@@ -128,39 +163,79 @@ static int stm32_dac_probe(struct platform_device *pdev)
priv->common.hfsel ? priv->common.hfsel ?
STM32H7_DAC_CR_HFSEL : 0); STM32H7_DAC_CR_HFSEL : 0);
if (ret) if (ret)
goto err_pclk; goto err_hw_stop;
} }
platform_set_drvdata(pdev, &priv->common);
ret = of_platform_populate(pdev->dev.of_node, NULL, NULL, dev); ret = of_platform_populate(pdev->dev.of_node, NULL, NULL, dev);
if (ret < 0) { if (ret < 0) {
dev_err(dev, "failed to populate DT children\n"); dev_err(dev, "failed to populate DT children\n");
goto err_pclk; goto err_hw_stop;
} }
pm_runtime_put(dev);
return 0; return 0;
err_pclk: err_hw_stop:
clk_disable_unprepare(priv->pclk); stm32_dac_core_hw_stop(dev);
err_vref: err_pm_stop:
regulator_disable(priv->vref); pm_runtime_disable(dev);
pm_runtime_set_suspended(dev);
pm_runtime_put_noidle(dev);
return ret; return ret;
} }
static int stm32_dac_remove(struct platform_device *pdev) static int stm32_dac_remove(struct platform_device *pdev)
{ {
struct stm32_dac_common *common = platform_get_drvdata(pdev); pm_runtime_get_sync(&pdev->dev);
of_platform_depopulate(&pdev->dev);
stm32_dac_core_hw_stop(&pdev->dev);
pm_runtime_disable(&pdev->dev);
pm_runtime_set_suspended(&pdev->dev);
pm_runtime_put_noidle(&pdev->dev);
return 0;
}
static int __maybe_unused stm32_dac_core_resume(struct device *dev)
{
struct stm32_dac_common *common = dev_get_drvdata(dev);
struct stm32_dac_priv *priv = to_stm32_dac_priv(common); struct stm32_dac_priv *priv = to_stm32_dac_priv(common);
int ret;
of_platform_depopulate(&pdev->dev); if (priv->common.hfsel) {
clk_disable_unprepare(priv->pclk); /* restore hfsel (maybe lost under low power state) */
regulator_disable(priv->vref); ret = regmap_update_bits(priv->common.regmap, STM32_DAC_CR,
STM32H7_DAC_CR_HFSEL,
STM32H7_DAC_CR_HFSEL);
if (ret)
return ret;
}
return pm_runtime_force_resume(dev);
}
static int __maybe_unused stm32_dac_core_runtime_suspend(struct device *dev)
{
stm32_dac_core_hw_stop(dev);
return 0; return 0;
} }
static int __maybe_unused stm32_dac_core_runtime_resume(struct device *dev)
{
return stm32_dac_core_hw_start(dev);
}
static const struct dev_pm_ops stm32_dac_core_pm_ops = {
SET_SYSTEM_SLEEP_PM_OPS(pm_runtime_force_suspend, stm32_dac_core_resume)
SET_RUNTIME_PM_OPS(stm32_dac_core_runtime_suspend,
stm32_dac_core_runtime_resume,
NULL)
};
static const struct stm32_dac_cfg stm32h7_dac_cfg = { static const struct stm32_dac_cfg stm32h7_dac_cfg = {
.has_hfsel = true, .has_hfsel = true,
}; };
...@@ -182,6 +257,7 @@ static struct platform_driver stm32_dac_driver = { ...@@ -182,6 +257,7 @@ static struct platform_driver stm32_dac_driver = {
.driver = { .driver = {
.name = "stm32-dac-core", .name = "stm32-dac-core",
.of_match_table = stm32_dac_of_match, .of_match_table = stm32_dac_of_match,
.pm = &stm32_dac_core_pm_ops,
}, },
}; };
module_platform_driver(stm32_dac_driver); module_platform_driver(stm32_dac_driver);
......
...@@ -13,6 +13,7 @@ ...@@ -13,6 +13,7 @@
#include <linux/kernel.h> #include <linux/kernel.h>
#include <linux/module.h> #include <linux/module.h>
#include <linux/platform_device.h> #include <linux/platform_device.h>
#include <linux/pm_runtime.h>
#include "stm32-dac-core.h" #include "stm32-dac-core.h"
...@@ -20,6 +21,8 @@ ...@@ -20,6 +21,8 @@
#define STM32_DAC_CHANNEL_2 2 #define STM32_DAC_CHANNEL_2 2
#define STM32_DAC_IS_CHAN_1(ch) ((ch) & STM32_DAC_CHANNEL_1) #define STM32_DAC_IS_CHAN_1(ch) ((ch) & STM32_DAC_CHANNEL_1)
#define STM32_DAC_AUTO_SUSPEND_DELAY_MS 2000
/** /**
* struct stm32_dac - private data of DAC driver * struct stm32_dac - private data of DAC driver
* @common: reference to DAC common data * @common: reference to DAC common data
...@@ -49,15 +52,34 @@ static int stm32_dac_set_enable_state(struct iio_dev *indio_dev, int ch, ...@@ -49,15 +52,34 @@ static int stm32_dac_set_enable_state(struct iio_dev *indio_dev, int ch,
bool enable) bool enable)
{ {
struct stm32_dac *dac = iio_priv(indio_dev); struct stm32_dac *dac = iio_priv(indio_dev);
struct device *dev = indio_dev->dev.parent;
u32 msk = STM32_DAC_IS_CHAN_1(ch) ? STM32_DAC_CR_EN1 : STM32_DAC_CR_EN2; u32 msk = STM32_DAC_IS_CHAN_1(ch) ? STM32_DAC_CR_EN1 : STM32_DAC_CR_EN2;
u32 en = enable ? msk : 0; u32 en = enable ? msk : 0;
int ret; int ret;
/* already enabled / disabled ? */
mutex_lock(&indio_dev->mlock);
ret = stm32_dac_is_enabled(indio_dev, ch);
if (ret < 0 || enable == !!ret) {
mutex_unlock(&indio_dev->mlock);
return ret < 0 ? ret : 0;
}
if (enable) {
ret = pm_runtime_get_sync(dev);
if (ret < 0) {
pm_runtime_put_noidle(dev);
mutex_unlock(&indio_dev->mlock);
return ret;
}
}
ret = regmap_update_bits(dac->common->regmap, STM32_DAC_CR, msk, en); ret = regmap_update_bits(dac->common->regmap, STM32_DAC_CR, msk, en);
mutex_unlock(&indio_dev->mlock);
if (ret < 0) { if (ret < 0) {
dev_err(&indio_dev->dev, "%s failed\n", en ? dev_err(&indio_dev->dev, "%s failed\n", en ?
"Enable" : "Disable"); "Enable" : "Disable");
return ret; goto err_put_pm;
} }
/* /*
...@@ -68,7 +90,20 @@ static int stm32_dac_set_enable_state(struct iio_dev *indio_dev, int ch, ...@@ -68,7 +90,20 @@ static int stm32_dac_set_enable_state(struct iio_dev *indio_dev, int ch,
if (en && dac->common->hfsel) if (en && dac->common->hfsel)
udelay(1); udelay(1);
if (!enable) {
pm_runtime_mark_last_busy(dev);
pm_runtime_put_autosuspend(dev);
}
return 0; return 0;
err_put_pm:
if (enable) {
pm_runtime_mark_last_busy(dev);
pm_runtime_put_autosuspend(dev);
}
return ret;
} }
static int stm32_dac_get_value(struct stm32_dac *dac, int channel, int *val) static int stm32_dac_get_value(struct stm32_dac *dac, int channel, int *val)
...@@ -272,6 +307,7 @@ static int stm32_dac_chan_of_init(struct iio_dev *indio_dev) ...@@ -272,6 +307,7 @@ static int stm32_dac_chan_of_init(struct iio_dev *indio_dev)
static int stm32_dac_probe(struct platform_device *pdev) static int stm32_dac_probe(struct platform_device *pdev)
{ {
struct device_node *np = pdev->dev.of_node; struct device_node *np = pdev->dev.of_node;
struct device *dev = &pdev->dev;
struct iio_dev *indio_dev; struct iio_dev *indio_dev;
struct stm32_dac *dac; struct stm32_dac *dac;
int ret; int ret;
...@@ -296,9 +332,61 @@ static int stm32_dac_probe(struct platform_device *pdev) ...@@ -296,9 +332,61 @@ static int stm32_dac_probe(struct platform_device *pdev)
if (ret < 0) if (ret < 0)
return ret; return ret;
return devm_iio_device_register(&pdev->dev, indio_dev); /* Get stm32-dac-core PM online */
pm_runtime_get_noresume(dev);
pm_runtime_set_active(dev);
pm_runtime_set_autosuspend_delay(dev, STM32_DAC_AUTO_SUSPEND_DELAY_MS);
pm_runtime_use_autosuspend(dev);
pm_runtime_enable(dev);
ret = iio_device_register(indio_dev);
if (ret)
goto err_pm_put;
pm_runtime_mark_last_busy(dev);
pm_runtime_put_autosuspend(dev);
return 0;
err_pm_put:
pm_runtime_disable(dev);
pm_runtime_set_suspended(dev);
pm_runtime_put_noidle(dev);
return ret;
}
static int stm32_dac_remove(struct platform_device *pdev)
{
struct iio_dev *indio_dev = platform_get_drvdata(pdev);
pm_runtime_get_sync(&pdev->dev);
iio_device_unregister(indio_dev);
pm_runtime_disable(&pdev->dev);
pm_runtime_set_suspended(&pdev->dev);
pm_runtime_put_noidle(&pdev->dev);
return 0;
} }
static int __maybe_unused stm32_dac_suspend(struct device *dev)
{
struct iio_dev *indio_dev = dev_get_drvdata(dev);
int channel = indio_dev->channels[0].channel;
int ret;
/* Ensure DAC is disabled before suspend */
ret = stm32_dac_is_enabled(indio_dev, channel);
if (ret)
return ret < 0 ? ret : -EBUSY;
return pm_runtime_force_suspend(dev);
}
static const struct dev_pm_ops stm32_dac_pm_ops = {
SET_SYSTEM_SLEEP_PM_OPS(stm32_dac_suspend, pm_runtime_force_resume)
};
static const struct of_device_id stm32_dac_of_match[] = { static const struct of_device_id stm32_dac_of_match[] = {
{ .compatible = "st,stm32-dac", }, { .compatible = "st,stm32-dac", },
{}, {},
...@@ -307,9 +395,11 @@ MODULE_DEVICE_TABLE(of, stm32_dac_of_match); ...@@ -307,9 +395,11 @@ MODULE_DEVICE_TABLE(of, stm32_dac_of_match);
static struct platform_driver stm32_dac_driver = { static struct platform_driver stm32_dac_driver = {
.probe = stm32_dac_probe, .probe = stm32_dac_probe,
.remove = stm32_dac_remove,
.driver = { .driver = {
.name = "stm32-dac", .name = "stm32-dac",
.of_match_table = stm32_dac_of_match, .of_match_table = stm32_dac_of_match,
.pm = &stm32_dac_pm_ops,
}, },
}; };
module_platform_driver(stm32_dac_driver); module_platform_driver(stm32_dac_driver);
......
...@@ -543,7 +543,7 @@ static irqreturn_t mpu3050_trigger_handler(int irq, void *p) ...@@ -543,7 +543,7 @@ static irqreturn_t mpu3050_trigger_handler(int irq, void *p)
toread = bytes_per_datum; toread = bytes_per_datum;
offset = 1; offset = 1;
/* Put in some dummy value */ /* Put in some dummy value */
fifo_values[0] = 0xAAAA; fifo_values[0] = cpu_to_be16(0xAAAA);
} }
ret = regmap_bulk_read(mpu3050->map, ret = regmap_bulk_read(mpu3050->map,
......
...@@ -14,7 +14,6 @@ ...@@ -14,7 +14,6 @@
#include <linux/types.h> #include <linux/types.h>
#include <linux/interrupt.h> #include <linux/interrupt.h>
#include <linux/i2c.h> #include <linux/i2c.h>
#include <linux/gpio.h>
#include <linux/irq.h> #include <linux/irq.h>
#include <linux/delay.h> #include <linux/delay.h>
#include <linux/iio/iio.h> #include <linux/iio/iio.h>
......
...@@ -40,6 +40,33 @@ config ADIS16480 ...@@ -40,6 +40,33 @@ config ADIS16480
source "drivers/iio/imu/bmi160/Kconfig" source "drivers/iio/imu/bmi160/Kconfig"
config FXOS8700
tristate
config FXOS8700_I2C
tristate "NXP FXOS8700 I2C driver"
depends on I2C
select FXOS8700
select REGMAP_I2C
help
Say yes here to build support for the NXP FXOS8700 m+g combo
sensor on I2C.
This driver can also be built as a module. If so, the module will be
called fxos8700_i2c.
config FXOS8700_SPI
tristate "NXP FXOS8700 SPI driver"
depends on SPI
select FXOS8700
select REGMAP_SPI
help
Say yes here to build support for the NXP FXOS8700 m+g combo
sensor on SPI.
This driver can also be built as a module. If so, the module will be
called fxos8700_spi.
config KMX61 config KMX61
tristate "Kionix KMX61 6-axis accelerometer and magnetometer" tristate "Kionix KMX61 6-axis accelerometer and magnetometer"
depends on I2C depends on I2C
......
...@@ -14,6 +14,11 @@ adis_lib-$(CONFIG_IIO_ADIS_LIB_BUFFER) += adis_buffer.o ...@@ -14,6 +14,11 @@ adis_lib-$(CONFIG_IIO_ADIS_LIB_BUFFER) += adis_buffer.o
obj-$(CONFIG_IIO_ADIS_LIB) += adis_lib.o obj-$(CONFIG_IIO_ADIS_LIB) += adis_lib.o
obj-y += bmi160/ obj-y += bmi160/
obj-$(CONFIG_FXOS8700) += fxos8700_core.o
obj-$(CONFIG_FXOS8700_I2C) += fxos8700_i2c.o
obj-$(CONFIG_FXOS8700_SPI) += fxos8700_spi.o
obj-y += inv_mpu6050/ obj-y += inv_mpu6050/
obj-$(CONFIG_KMX61) += kmx61.o obj-$(CONFIG_KMX61) += kmx61.o
......
/* SPDX-License-Identifier: GPL-2.0 */
#ifndef FXOS8700_H_
#define FXOS8700_H_
extern const struct regmap_config fxos8700_regmap_config;
int fxos8700_core_probe(struct device *dev, struct regmap *regmap,
const char *name, bool use_spi);
#endif /* FXOS8700_H_ */
This diff is collapsed.
This diff is collapsed.
// SPDX-License-Identifier: GPL-2.0
/*
* FXOS8700 - NXP IMU, SPI bits
*/
#include <linux/acpi.h>
#include <linux/module.h>
#include <linux/mod_devicetable.h>
#include <linux/regmap.h>
#include <linux/spi/spi.h>
#include "fxos8700.h"
static int fxos8700_spi_probe(struct spi_device *spi)
{
struct regmap *regmap;
const struct spi_device_id *id = spi_get_device_id(spi);
regmap = devm_regmap_init_spi(spi, &fxos8700_regmap_config);
if (IS_ERR(regmap)) {
dev_err(&spi->dev, "Failed to register spi regmap %d\n",
(int)PTR_ERR(regmap));
return PTR_ERR(regmap);
}
return fxos8700_core_probe(&spi->dev, regmap, id->name, true);
}
static const struct spi_device_id fxos8700_spi_id[] = {
{"fxos8700", 0},
{ }
};
MODULE_DEVICE_TABLE(spi, fxos8700_spi_id);
static const struct acpi_device_id fxos8700_acpi_match[] = {
{"FXOS8700", 0},
{ }
};
MODULE_DEVICE_TABLE(acpi, fxos8700_acpi_match);
static const struct of_device_id fxos8700_of_match[] = {
{ .compatible = "nxp,fxos8700" },
{ }
};
MODULE_DEVICE_TABLE(of, fxos8700_of_match);
static struct spi_driver fxos8700_spi_driver = {
.probe = fxos8700_spi_probe,
.id_table = fxos8700_spi_id,
.driver = {
.acpi_match_table = ACPI_PTR(fxos8700_acpi_match),
.of_match_table = fxos8700_of_match,
.name = "fxos8700_spi",
},
};
module_spi_driver(fxos8700_spi_driver);
MODULE_AUTHOR("Robert Jones <rjones@gateworks.com>");
MODULE_DESCRIPTION("FXOS8700 SPI driver");
MODULE_LICENSE("GPL v2");
...@@ -12,7 +12,8 @@ config IIO_ST_LSM6DSX ...@@ -12,7 +12,8 @@ config IIO_ST_LSM6DSX
Say yes here to build support for STMicroelectronics LSM6DSx imu Say yes here to build support for STMicroelectronics LSM6DSx imu
sensor. Supported devices: lsm6ds3, lsm6ds3h, lsm6dsl, lsm6dsm, sensor. Supported devices: lsm6ds3, lsm6ds3h, lsm6dsl, lsm6dsm,
ism330dlc, lsm6dso, lsm6dsox, asm330lhh, lsm6dsr, lsm6ds3tr-c, ism330dlc, lsm6dso, lsm6dsox, asm330lhh, lsm6dsr, lsm6ds3tr-c,
ism330dhcx and the accelerometer/gyroscope of lsm9ds1. ism330dhcx, lsm6dsrx, lsm6ds0 and the accelerometer/gyroscope
of lsm9ds1.
To compile this driver as a module, choose M here: the module To compile this driver as a module, choose M here: the module
will be called st_lsm6dsx. will be called st_lsm6dsx.
......
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
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