1*4882a593Smuzhiyun /* SPDX-License-Identifier: GPL-2.0-only */
2*4882a593Smuzhiyun /* Copyright (c) 2012-2013, The Linux Foundation. All rights reserved.
3*4882a593Smuzhiyun */
4*4882a593Smuzhiyun #ifndef _LINUX_SPMI_H
5*4882a593Smuzhiyun #define _LINUX_SPMI_H
6*4882a593Smuzhiyun
7*4882a593Smuzhiyun #include <linux/types.h>
8*4882a593Smuzhiyun #include <linux/device.h>
9*4882a593Smuzhiyun #include <linux/mod_devicetable.h>
10*4882a593Smuzhiyun #include <linux/android_kabi.h>
11*4882a593Smuzhiyun
12*4882a593Smuzhiyun /* Maximum slave identifier */
13*4882a593Smuzhiyun #define SPMI_MAX_SLAVE_ID 16
14*4882a593Smuzhiyun
15*4882a593Smuzhiyun /* SPMI Commands */
16*4882a593Smuzhiyun #define SPMI_CMD_EXT_WRITE 0x00
17*4882a593Smuzhiyun #define SPMI_CMD_RESET 0x10
18*4882a593Smuzhiyun #define SPMI_CMD_SLEEP 0x11
19*4882a593Smuzhiyun #define SPMI_CMD_SHUTDOWN 0x12
20*4882a593Smuzhiyun #define SPMI_CMD_WAKEUP 0x13
21*4882a593Smuzhiyun #define SPMI_CMD_AUTHENTICATE 0x14
22*4882a593Smuzhiyun #define SPMI_CMD_MSTR_READ 0x15
23*4882a593Smuzhiyun #define SPMI_CMD_MSTR_WRITE 0x16
24*4882a593Smuzhiyun #define SPMI_CMD_TRANSFER_BUS_OWNERSHIP 0x1A
25*4882a593Smuzhiyun #define SPMI_CMD_DDB_MASTER_READ 0x1B
26*4882a593Smuzhiyun #define SPMI_CMD_DDB_SLAVE_READ 0x1C
27*4882a593Smuzhiyun #define SPMI_CMD_EXT_READ 0x20
28*4882a593Smuzhiyun #define SPMI_CMD_EXT_WRITEL 0x30
29*4882a593Smuzhiyun #define SPMI_CMD_EXT_READL 0x38
30*4882a593Smuzhiyun #define SPMI_CMD_WRITE 0x40
31*4882a593Smuzhiyun #define SPMI_CMD_READ 0x60
32*4882a593Smuzhiyun #define SPMI_CMD_ZERO_WRITE 0x80
33*4882a593Smuzhiyun
34*4882a593Smuzhiyun /**
35*4882a593Smuzhiyun * struct spmi_device - Basic representation of an SPMI device
36*4882a593Smuzhiyun * @dev: Driver model representation of the device.
37*4882a593Smuzhiyun * @ctrl: SPMI controller managing the bus hosting this device.
38*4882a593Smuzhiyun * @usid: This devices' Unique Slave IDentifier.
39*4882a593Smuzhiyun */
40*4882a593Smuzhiyun struct spmi_device {
41*4882a593Smuzhiyun struct device dev;
42*4882a593Smuzhiyun struct spmi_controller *ctrl;
43*4882a593Smuzhiyun u8 usid;
44*4882a593Smuzhiyun };
45*4882a593Smuzhiyun
to_spmi_device(struct device * d)46*4882a593Smuzhiyun static inline struct spmi_device *to_spmi_device(struct device *d)
47*4882a593Smuzhiyun {
48*4882a593Smuzhiyun return container_of(d, struct spmi_device, dev);
49*4882a593Smuzhiyun }
50*4882a593Smuzhiyun
spmi_device_get_drvdata(const struct spmi_device * sdev)51*4882a593Smuzhiyun static inline void *spmi_device_get_drvdata(const struct spmi_device *sdev)
52*4882a593Smuzhiyun {
53*4882a593Smuzhiyun return dev_get_drvdata(&sdev->dev);
54*4882a593Smuzhiyun }
55*4882a593Smuzhiyun
spmi_device_set_drvdata(struct spmi_device * sdev,void * data)56*4882a593Smuzhiyun static inline void spmi_device_set_drvdata(struct spmi_device *sdev, void *data)
57*4882a593Smuzhiyun {
58*4882a593Smuzhiyun dev_set_drvdata(&sdev->dev, data);
59*4882a593Smuzhiyun }
60*4882a593Smuzhiyun
61*4882a593Smuzhiyun struct spmi_device *spmi_device_alloc(struct spmi_controller *ctrl);
62*4882a593Smuzhiyun
spmi_device_put(struct spmi_device * sdev)63*4882a593Smuzhiyun static inline void spmi_device_put(struct spmi_device *sdev)
64*4882a593Smuzhiyun {
65*4882a593Smuzhiyun if (sdev)
66*4882a593Smuzhiyun put_device(&sdev->dev);
67*4882a593Smuzhiyun }
68*4882a593Smuzhiyun
69*4882a593Smuzhiyun int spmi_device_add(struct spmi_device *sdev);
70*4882a593Smuzhiyun
71*4882a593Smuzhiyun void spmi_device_remove(struct spmi_device *sdev);
72*4882a593Smuzhiyun
73*4882a593Smuzhiyun /**
74*4882a593Smuzhiyun * struct spmi_controller - interface to the SPMI master controller
75*4882a593Smuzhiyun * @dev: Driver model representation of the device.
76*4882a593Smuzhiyun * @nr: board-specific number identifier for this controller/bus
77*4882a593Smuzhiyun * @cmd: sends a non-data command sequence on the SPMI bus.
78*4882a593Smuzhiyun * @read_cmd: sends a register read command sequence on the SPMI bus.
79*4882a593Smuzhiyun * @write_cmd: sends a register write command sequence on the SPMI bus.
80*4882a593Smuzhiyun */
81*4882a593Smuzhiyun struct spmi_controller {
82*4882a593Smuzhiyun struct device dev;
83*4882a593Smuzhiyun unsigned int nr;
84*4882a593Smuzhiyun int (*cmd)(struct spmi_controller *ctrl, u8 opcode, u8 sid);
85*4882a593Smuzhiyun int (*read_cmd)(struct spmi_controller *ctrl, u8 opcode,
86*4882a593Smuzhiyun u8 sid, u16 addr, u8 *buf, size_t len);
87*4882a593Smuzhiyun int (*write_cmd)(struct spmi_controller *ctrl, u8 opcode,
88*4882a593Smuzhiyun u8 sid, u16 addr, const u8 *buf, size_t len);
89*4882a593Smuzhiyun ANDROID_KABI_RESERVE(1);
90*4882a593Smuzhiyun };
91*4882a593Smuzhiyun
to_spmi_controller(struct device * d)92*4882a593Smuzhiyun static inline struct spmi_controller *to_spmi_controller(struct device *d)
93*4882a593Smuzhiyun {
94*4882a593Smuzhiyun return container_of(d, struct spmi_controller, dev);
95*4882a593Smuzhiyun }
96*4882a593Smuzhiyun
97*4882a593Smuzhiyun static inline
spmi_controller_get_drvdata(const struct spmi_controller * ctrl)98*4882a593Smuzhiyun void *spmi_controller_get_drvdata(const struct spmi_controller *ctrl)
99*4882a593Smuzhiyun {
100*4882a593Smuzhiyun return dev_get_drvdata(&ctrl->dev);
101*4882a593Smuzhiyun }
102*4882a593Smuzhiyun
spmi_controller_set_drvdata(struct spmi_controller * ctrl,void * data)103*4882a593Smuzhiyun static inline void spmi_controller_set_drvdata(struct spmi_controller *ctrl,
104*4882a593Smuzhiyun void *data)
105*4882a593Smuzhiyun {
106*4882a593Smuzhiyun dev_set_drvdata(&ctrl->dev, data);
107*4882a593Smuzhiyun }
108*4882a593Smuzhiyun
109*4882a593Smuzhiyun struct spmi_controller *spmi_controller_alloc(struct device *parent,
110*4882a593Smuzhiyun size_t size);
111*4882a593Smuzhiyun
112*4882a593Smuzhiyun /**
113*4882a593Smuzhiyun * spmi_controller_put() - decrement controller refcount
114*4882a593Smuzhiyun * @ctrl SPMI controller.
115*4882a593Smuzhiyun */
spmi_controller_put(struct spmi_controller * ctrl)116*4882a593Smuzhiyun static inline void spmi_controller_put(struct spmi_controller *ctrl)
117*4882a593Smuzhiyun {
118*4882a593Smuzhiyun if (ctrl)
119*4882a593Smuzhiyun put_device(&ctrl->dev);
120*4882a593Smuzhiyun }
121*4882a593Smuzhiyun
122*4882a593Smuzhiyun int spmi_controller_add(struct spmi_controller *ctrl);
123*4882a593Smuzhiyun void spmi_controller_remove(struct spmi_controller *ctrl);
124*4882a593Smuzhiyun
125*4882a593Smuzhiyun /**
126*4882a593Smuzhiyun * struct spmi_driver - SPMI slave device driver
127*4882a593Smuzhiyun * @driver: SPMI device drivers should initialize name and owner field of
128*4882a593Smuzhiyun * this structure.
129*4882a593Smuzhiyun * @probe: binds this driver to a SPMI device.
130*4882a593Smuzhiyun * @remove: unbinds this driver from the SPMI device.
131*4882a593Smuzhiyun *
132*4882a593Smuzhiyun * If PM runtime support is desired for a slave, a device driver can call
133*4882a593Smuzhiyun * pm_runtime_put() from their probe() routine (and a balancing
134*4882a593Smuzhiyun * pm_runtime_get() in remove()). PM runtime support for a slave is
135*4882a593Smuzhiyun * implemented by issuing a SLEEP command to the slave on runtime_suspend(),
136*4882a593Smuzhiyun * transitioning the slave into the SLEEP state. On runtime_resume(), a WAKEUP
137*4882a593Smuzhiyun * command is sent to the slave to bring it back to ACTIVE.
138*4882a593Smuzhiyun */
139*4882a593Smuzhiyun struct spmi_driver {
140*4882a593Smuzhiyun struct device_driver driver;
141*4882a593Smuzhiyun int (*probe)(struct spmi_device *sdev);
142*4882a593Smuzhiyun void (*remove)(struct spmi_device *sdev);
143*4882a593Smuzhiyun ANDROID_KABI_RESERVE(1);
144*4882a593Smuzhiyun };
145*4882a593Smuzhiyun
to_spmi_driver(struct device_driver * d)146*4882a593Smuzhiyun static inline struct spmi_driver *to_spmi_driver(struct device_driver *d)
147*4882a593Smuzhiyun {
148*4882a593Smuzhiyun return container_of(d, struct spmi_driver, driver);
149*4882a593Smuzhiyun }
150*4882a593Smuzhiyun
151*4882a593Smuzhiyun #define spmi_driver_register(sdrv) \
152*4882a593Smuzhiyun __spmi_driver_register(sdrv, THIS_MODULE)
153*4882a593Smuzhiyun int __spmi_driver_register(struct spmi_driver *sdrv, struct module *owner);
154*4882a593Smuzhiyun
155*4882a593Smuzhiyun /**
156*4882a593Smuzhiyun * spmi_driver_unregister() - unregister an SPMI client driver
157*4882a593Smuzhiyun * @sdrv: the driver to unregister
158*4882a593Smuzhiyun */
spmi_driver_unregister(struct spmi_driver * sdrv)159*4882a593Smuzhiyun static inline void spmi_driver_unregister(struct spmi_driver *sdrv)
160*4882a593Smuzhiyun {
161*4882a593Smuzhiyun if (sdrv)
162*4882a593Smuzhiyun driver_unregister(&sdrv->driver);
163*4882a593Smuzhiyun }
164*4882a593Smuzhiyun
165*4882a593Smuzhiyun #define module_spmi_driver(__spmi_driver) \
166*4882a593Smuzhiyun module_driver(__spmi_driver, spmi_driver_register, \
167*4882a593Smuzhiyun spmi_driver_unregister)
168*4882a593Smuzhiyun
169*4882a593Smuzhiyun int spmi_register_read(struct spmi_device *sdev, u8 addr, u8 *buf);
170*4882a593Smuzhiyun int spmi_ext_register_read(struct spmi_device *sdev, u8 addr, u8 *buf,
171*4882a593Smuzhiyun size_t len);
172*4882a593Smuzhiyun int spmi_ext_register_readl(struct spmi_device *sdev, u16 addr, u8 *buf,
173*4882a593Smuzhiyun size_t len);
174*4882a593Smuzhiyun int spmi_register_write(struct spmi_device *sdev, u8 addr, u8 data);
175*4882a593Smuzhiyun int spmi_register_zero_write(struct spmi_device *sdev, u8 data);
176*4882a593Smuzhiyun int spmi_ext_register_write(struct spmi_device *sdev, u8 addr,
177*4882a593Smuzhiyun const u8 *buf, size_t len);
178*4882a593Smuzhiyun int spmi_ext_register_writel(struct spmi_device *sdev, u16 addr,
179*4882a593Smuzhiyun const u8 *buf, size_t len);
180*4882a593Smuzhiyun int spmi_command_reset(struct spmi_device *sdev);
181*4882a593Smuzhiyun int spmi_command_sleep(struct spmi_device *sdev);
182*4882a593Smuzhiyun int spmi_command_wakeup(struct spmi_device *sdev);
183*4882a593Smuzhiyun int spmi_command_shutdown(struct spmi_device *sdev);
184*4882a593Smuzhiyun
185*4882a593Smuzhiyun #endif
186