xref: /OK3568_Linux_fs/kernel/drivers/input/misc/adxl34x-spi.c (revision 4882a59341e53eb6f0b4789bf948001014eff981)
1*4882a593Smuzhiyun // SPDX-License-Identifier: GPL-2.0-or-later
2*4882a593Smuzhiyun /*
3*4882a593Smuzhiyun  * ADLX345/346 Three-Axis Digital Accelerometers (SPI Interface)
4*4882a593Smuzhiyun  *
5*4882a593Smuzhiyun  * Enter bugs at http://blackfin.uclinux.org/
6*4882a593Smuzhiyun  *
7*4882a593Smuzhiyun  * Copyright (C) 2009 Michael Hennerich, Analog Devices Inc.
8*4882a593Smuzhiyun  */
9*4882a593Smuzhiyun 
10*4882a593Smuzhiyun #include <linux/input.h>	/* BUS_SPI */
11*4882a593Smuzhiyun #include <linux/module.h>
12*4882a593Smuzhiyun #include <linux/spi/spi.h>
13*4882a593Smuzhiyun #include <linux/pm.h>
14*4882a593Smuzhiyun #include <linux/types.h>
15*4882a593Smuzhiyun #include "adxl34x.h"
16*4882a593Smuzhiyun 
17*4882a593Smuzhiyun #define MAX_SPI_FREQ_HZ		5000000
18*4882a593Smuzhiyun #define MAX_FREQ_NO_FIFODELAY	1500000
19*4882a593Smuzhiyun #define ADXL34X_CMD_MULTB	(1 << 6)
20*4882a593Smuzhiyun #define ADXL34X_CMD_READ	(1 << 7)
21*4882a593Smuzhiyun #define ADXL34X_WRITECMD(reg)	(reg & 0x3F)
22*4882a593Smuzhiyun #define ADXL34X_READCMD(reg)	(ADXL34X_CMD_READ | (reg & 0x3F))
23*4882a593Smuzhiyun #define ADXL34X_READMB_CMD(reg) (ADXL34X_CMD_READ | ADXL34X_CMD_MULTB \
24*4882a593Smuzhiyun 					| (reg & 0x3F))
25*4882a593Smuzhiyun 
adxl34x_spi_read(struct device * dev,unsigned char reg)26*4882a593Smuzhiyun static int adxl34x_spi_read(struct device *dev, unsigned char reg)
27*4882a593Smuzhiyun {
28*4882a593Smuzhiyun 	struct spi_device *spi = to_spi_device(dev);
29*4882a593Smuzhiyun 	unsigned char cmd;
30*4882a593Smuzhiyun 
31*4882a593Smuzhiyun 	cmd = ADXL34X_READCMD(reg);
32*4882a593Smuzhiyun 
33*4882a593Smuzhiyun 	return spi_w8r8(spi, cmd);
34*4882a593Smuzhiyun }
35*4882a593Smuzhiyun 
adxl34x_spi_write(struct device * dev,unsigned char reg,unsigned char val)36*4882a593Smuzhiyun static int adxl34x_spi_write(struct device *dev,
37*4882a593Smuzhiyun 			     unsigned char reg, unsigned char val)
38*4882a593Smuzhiyun {
39*4882a593Smuzhiyun 	struct spi_device *spi = to_spi_device(dev);
40*4882a593Smuzhiyun 	unsigned char buf[2];
41*4882a593Smuzhiyun 
42*4882a593Smuzhiyun 	buf[0] = ADXL34X_WRITECMD(reg);
43*4882a593Smuzhiyun 	buf[1] = val;
44*4882a593Smuzhiyun 
45*4882a593Smuzhiyun 	return spi_write(spi, buf, sizeof(buf));
46*4882a593Smuzhiyun }
47*4882a593Smuzhiyun 
adxl34x_spi_read_block(struct device * dev,unsigned char reg,int count,void * buf)48*4882a593Smuzhiyun static int adxl34x_spi_read_block(struct device *dev,
49*4882a593Smuzhiyun 				  unsigned char reg, int count,
50*4882a593Smuzhiyun 				  void *buf)
51*4882a593Smuzhiyun {
52*4882a593Smuzhiyun 	struct spi_device *spi = to_spi_device(dev);
53*4882a593Smuzhiyun 	ssize_t status;
54*4882a593Smuzhiyun 
55*4882a593Smuzhiyun 	reg = ADXL34X_READMB_CMD(reg);
56*4882a593Smuzhiyun 	status = spi_write_then_read(spi, &reg, 1, buf, count);
57*4882a593Smuzhiyun 
58*4882a593Smuzhiyun 	return (status < 0) ? status : 0;
59*4882a593Smuzhiyun }
60*4882a593Smuzhiyun 
61*4882a593Smuzhiyun static const struct adxl34x_bus_ops adxl34x_spi_bops = {
62*4882a593Smuzhiyun 	.bustype	= BUS_SPI,
63*4882a593Smuzhiyun 	.write		= adxl34x_spi_write,
64*4882a593Smuzhiyun 	.read		= adxl34x_spi_read,
65*4882a593Smuzhiyun 	.read_block	= adxl34x_spi_read_block,
66*4882a593Smuzhiyun };
67*4882a593Smuzhiyun 
adxl34x_spi_probe(struct spi_device * spi)68*4882a593Smuzhiyun static int adxl34x_spi_probe(struct spi_device *spi)
69*4882a593Smuzhiyun {
70*4882a593Smuzhiyun 	struct adxl34x *ac;
71*4882a593Smuzhiyun 
72*4882a593Smuzhiyun 	/* don't exceed max specified SPI CLK frequency */
73*4882a593Smuzhiyun 	if (spi->max_speed_hz > MAX_SPI_FREQ_HZ) {
74*4882a593Smuzhiyun 		dev_err(&spi->dev, "SPI CLK %d Hz too fast\n", spi->max_speed_hz);
75*4882a593Smuzhiyun 		return -EINVAL;
76*4882a593Smuzhiyun 	}
77*4882a593Smuzhiyun 
78*4882a593Smuzhiyun 	ac = adxl34x_probe(&spi->dev, spi->irq,
79*4882a593Smuzhiyun 			   spi->max_speed_hz > MAX_FREQ_NO_FIFODELAY,
80*4882a593Smuzhiyun 			   &adxl34x_spi_bops);
81*4882a593Smuzhiyun 
82*4882a593Smuzhiyun 	if (IS_ERR(ac))
83*4882a593Smuzhiyun 		return PTR_ERR(ac);
84*4882a593Smuzhiyun 
85*4882a593Smuzhiyun 	spi_set_drvdata(spi, ac);
86*4882a593Smuzhiyun 
87*4882a593Smuzhiyun 	return 0;
88*4882a593Smuzhiyun }
89*4882a593Smuzhiyun 
adxl34x_spi_remove(struct spi_device * spi)90*4882a593Smuzhiyun static int adxl34x_spi_remove(struct spi_device *spi)
91*4882a593Smuzhiyun {
92*4882a593Smuzhiyun 	struct adxl34x *ac = spi_get_drvdata(spi);
93*4882a593Smuzhiyun 
94*4882a593Smuzhiyun 	return adxl34x_remove(ac);
95*4882a593Smuzhiyun }
96*4882a593Smuzhiyun 
adxl34x_spi_suspend(struct device * dev)97*4882a593Smuzhiyun static int __maybe_unused adxl34x_spi_suspend(struct device *dev)
98*4882a593Smuzhiyun {
99*4882a593Smuzhiyun 	struct spi_device *spi = to_spi_device(dev);
100*4882a593Smuzhiyun 	struct adxl34x *ac = spi_get_drvdata(spi);
101*4882a593Smuzhiyun 
102*4882a593Smuzhiyun 	adxl34x_suspend(ac);
103*4882a593Smuzhiyun 
104*4882a593Smuzhiyun 	return 0;
105*4882a593Smuzhiyun }
106*4882a593Smuzhiyun 
adxl34x_spi_resume(struct device * dev)107*4882a593Smuzhiyun static int __maybe_unused adxl34x_spi_resume(struct device *dev)
108*4882a593Smuzhiyun {
109*4882a593Smuzhiyun 	struct spi_device *spi = to_spi_device(dev);
110*4882a593Smuzhiyun 	struct adxl34x *ac = spi_get_drvdata(spi);
111*4882a593Smuzhiyun 
112*4882a593Smuzhiyun 	adxl34x_resume(ac);
113*4882a593Smuzhiyun 
114*4882a593Smuzhiyun 	return 0;
115*4882a593Smuzhiyun }
116*4882a593Smuzhiyun 
117*4882a593Smuzhiyun static SIMPLE_DEV_PM_OPS(adxl34x_spi_pm, adxl34x_spi_suspend,
118*4882a593Smuzhiyun 			 adxl34x_spi_resume);
119*4882a593Smuzhiyun 
120*4882a593Smuzhiyun static struct spi_driver adxl34x_driver = {
121*4882a593Smuzhiyun 	.driver = {
122*4882a593Smuzhiyun 		.name = "adxl34x",
123*4882a593Smuzhiyun 		.pm = &adxl34x_spi_pm,
124*4882a593Smuzhiyun 	},
125*4882a593Smuzhiyun 	.probe   = adxl34x_spi_probe,
126*4882a593Smuzhiyun 	.remove  = adxl34x_spi_remove,
127*4882a593Smuzhiyun };
128*4882a593Smuzhiyun 
129*4882a593Smuzhiyun module_spi_driver(adxl34x_driver);
130*4882a593Smuzhiyun 
131*4882a593Smuzhiyun MODULE_AUTHOR("Michael Hennerich <hennerich@blackfin.uclinux.org>");
132*4882a593Smuzhiyun MODULE_DESCRIPTION("ADXL345/346 Three-Axis Digital Accelerometer SPI Bus Driver");
133*4882a593Smuzhiyun MODULE_LICENSE("GPL");
134