2 * SPI slave handler controlling system state
4 * This SPI slave handler allows remote control of system reboot, power off,
7 * Copyright (C) 2016-2017 Glider bvba
9 * This file is subject to the terms and conditions of the GNU General Public
10 * License. See the file "COPYING" in the main directory of this archive
13 * Usage (assuming /dev/spidev2.0 corresponds to the SPI master on the remote
17 * # poweroff='\x71\x3f'
19 * # suspend='\x1b\x1b'
20 * # spidev_test -D /dev/spidev2.0 -p $suspend # or $reboot, $poweroff, $halt
23 #include <linux/completion.h>
24 #include <linux/module.h>
25 #include <linux/reboot.h>
26 #include <linux/suspend.h>
27 #include <linux/spi/spi.h>
30 * The numbers are chosen to display something human-readable on two 7-segment
31 * displays connected to two 74HC595 shift registers
33 #define CMD_REBOOT 0x7c50 /* rb */
34 #define CMD_POWEROFF 0x713f /* OF */
35 #define CMD_HALT 0x3876 /* HL */
36 #define CMD_SUSPEND 0x1b1b /* ZZ */
38 struct spi_slave_system_control_priv
{
39 struct spi_device
*spi
;
40 struct completion finished
;
41 struct spi_transfer xfer
;
42 struct spi_message msg
;
47 int spi_slave_system_control_submit(struct spi_slave_system_control_priv
*priv
);
49 static void spi_slave_system_control_complete(void *arg
)
51 struct spi_slave_system_control_priv
*priv
= arg
;
58 cmd
= be16_to_cpu(priv
->cmd
);
61 dev_info(&priv
->spi
->dev
, "Rebooting system...\n");
66 dev_info(&priv
->spi
->dev
, "Powering off system...\n");
71 dev_info(&priv
->spi
->dev
, "Halting system...\n");
76 dev_info(&priv
->spi
->dev
, "Suspending system...\n");
77 pm_suspend(PM_SUSPEND_MEM
);
81 dev_warn(&priv
->spi
->dev
, "Unknown command 0x%x\n", cmd
);
85 ret
= spi_slave_system_control_submit(priv
);
92 dev_info(&priv
->spi
->dev
, "Terminating\n");
93 complete(&priv
->finished
);
97 int spi_slave_system_control_submit(struct spi_slave_system_control_priv
*priv
)
101 spi_message_init_with_transfers(&priv
->msg
, &priv
->xfer
, 1);
103 priv
->msg
.complete
= spi_slave_system_control_complete
;
104 priv
->msg
.context
= priv
;
106 ret
= spi_async(priv
->spi
, &priv
->msg
);
108 dev_err(&priv
->spi
->dev
, "spi_async() failed %d\n", ret
);
113 static int spi_slave_system_control_probe(struct spi_device
*spi
)
115 struct spi_slave_system_control_priv
*priv
;
118 priv
= devm_kzalloc(&spi
->dev
, sizeof(*priv
), GFP_KERNEL
);
123 init_completion(&priv
->finished
);
124 priv
->xfer
.rx_buf
= &priv
->cmd
;
125 priv
->xfer
.len
= sizeof(priv
->cmd
);
127 ret
= spi_slave_system_control_submit(priv
);
131 spi_set_drvdata(spi
, priv
);
135 static void spi_slave_system_control_remove(struct spi_device
*spi
)
137 struct spi_slave_system_control_priv
*priv
= spi_get_drvdata(spi
);
139 spi_target_abort(spi
);
140 wait_for_completion(&priv
->finished
);
143 static struct spi_driver spi_slave_system_control_driver
= {
145 .name
= "spi-slave-system-control",
147 .probe
= spi_slave_system_control_probe
,
148 .remove
= spi_slave_system_control_remove
,
150 module_spi_driver(spi_slave_system_control_driver
);
152 MODULE_AUTHOR("Geert Uytterhoeven <geert+renesas@glider.be>");
153 MODULE_DESCRIPTION("SPI slave handler controlling system state");
154 MODULE_LICENSE("GPL v2");