The Thecus N1200 has a board control PIC on the I2C bus. This controls two of the LEDs, the buzzer and allows the board to poweroff.
Signed-off-by: Byron Bradley <byron.bbrad...@gmail.com> --- arch/powerpc/configs/83xx/thecus_n1200_defconfig | 3 +- arch/powerpc/platforms/83xx/Makefile | 1 + arch/powerpc/platforms/83xx/mcu_thecus_n1200.c | 307 ++++++++++++++++++++++ arch/powerpc/platforms/Kconfig | 9 + 4 files changed, 319 insertions(+), 1 deletions(-) create mode 100644 arch/powerpc/platforms/83xx/mcu_thecus_n1200.c diff --git a/arch/powerpc/configs/83xx/thecus_n1200_defconfig b/arch/powerpc/configs/83xx/thecus_n1200_defconfig index b0c3a15..a0731e6 100644 --- a/arch/powerpc/configs/83xx/thecus_n1200_defconfig +++ b/arch/powerpc/configs/83xx/thecus_n1200_defconfig @@ -1,7 +1,7 @@ # # Automatically generated make config: don't edit # Linux kernel version: 2.6.30-rc6 -# Fri May 22 18:36:51 2009 +# Fri May 22 19:01:15 2009 # # CONFIG_PPC64 is not set @@ -217,6 +217,7 @@ CONFIG_IPIC=y CONFIG_MPC8xxx_GPIO=y # CONFIG_SIMPLE_GPIO is not set # CONFIG_MCU_MPC8349EMITX is not set +CONFIG_MCU_THECUS_N1200=m # # Kernel options diff --git a/arch/powerpc/platforms/83xx/Makefile b/arch/powerpc/platforms/83xx/Makefile index 2f2d522..682797c 100644 --- a/arch/powerpc/platforms/83xx/Makefile +++ b/arch/powerpc/platforms/83xx/Makefile @@ -15,4 +15,5 @@ obj-$(CONFIG_MPC837x_MDS) += mpc837x_mds.o obj-$(CONFIG_SBC834x) += sbc834x.o obj-$(CONFIG_MPC837x_RDB) += mpc837x_rdb.o obj-$(CONFIG_THECUS_N1200) += thecus_n1200.o +obj-$(CONFIG_MCU_THECUS_N1200) += mcu_thecus_n1200.o obj-$(CONFIG_ASP834x) += asp834x.o diff --git a/arch/powerpc/platforms/83xx/mcu_thecus_n1200.c b/arch/powerpc/platforms/83xx/mcu_thecus_n1200.c new file mode 100644 index 0000000..03384ce --- /dev/null +++ b/arch/powerpc/platforms/83xx/mcu_thecus_n1200.c @@ -0,0 +1,307 @@ +/* + * Driver for the MCU on Thecus N1200 boards. + * + * Copyright (c) 2009 Byron Bradley + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License + * as published by the Free Software Foundation; either version + * 2 of the License, or (at your option) any later version. + */ + +/* + * The Thecus N1200 boards have a board control PIC on the I2C bus. This + * provides control over two of the LEDs, buzzer and the power-off function. + * The behaviour of the power button changes depending on the power state so we + * tell the PIC we have booted when the module loads and tell it when we are + * rebooting. + */ + +#include <asm/machdep.h> +#include <linux/delay.h> +#include <linux/device.h> +#include <linux/i2c.h> +#include <linux/input.h> +#include <linux/leds.h> +#include <linux/module.h> +#include <linux/reboot.h> + +/* N1200 registers */ +#define N1200_REG_PWR_CTRL 0x0 +#define N1200_REG_LED_SATA 0x1 +#define N1200_REG_LED_SYSTEM 0x2 +#define N1200_REG_LED_BUSY 0x3 +#define N1200_REG_BUZZER 0x4 +#define N1200_REG_PWR_STATUS 0x5 +#define N1200_REG_VERSION 0x6 + +/* for N1200_REG_PWR_CTRL */ +#define N1200_PWR_OFF 0x0 +#define N1200_PWR_RESET 0x1 + +/* for N1200_REG_PWR_STATUS */ +#define N1200_PWR_OFF 0x0 +#define N1200_PWR_BOOTED 0x2 + +/* for N1200_REG_LED_* */ +#define N1200_LED_OFF 0x0 +#define N1200_LED_ON 0x1 +#define N1200_LED_BLINK 0x2 + +/* We need to store a pointer to the i2c_client for use during a poweroff */ +static struct i2c_client *n1200_i2c_client; + +struct mcu_n1200 { + struct led_classdev sata_led; + struct led_classdev system_led; + struct led_classdev busy_led; + struct input_dev *input_dev; +}; + +static inline int n1200_read(struct i2c_client *client, u8 reg) +{ + return i2c_smbus_read_byte_data(client, reg); +} + +static inline void n1200_write(struct i2c_client *client, u8 reg, u8 value) +{ + i2c_smbus_write_byte_data(client, reg, value); +} + +void n1200_power_off(void) +{ + if (n1200_i2c_client == NULL) + return; + + n1200_write(n1200_i2c_client, N1200_REG_PWR_CTRL, N1200_PWR_OFF); +} + +static int n1200_reboot_notify(struct notifier_block *nb, unsigned long event, + void *p) +{ + if (event != SYS_RESTART || n1200_i2c_client == NULL) + return NOTIFY_DONE; + + n1200_write(n1200_i2c_client, N1200_REG_PWR_CTRL, N1200_PWR_RESET); + + return NOTIFY_DONE; +} + +struct notifier_block n1200_notifier = { + .notifier_call = n1200_reboot_notify, +}; + +#define N1200_NEW_LED(__name, __reg, __blinkable) \ +static void n1200_##__name##_set(struct led_classdev *led_cdev, \ + enum led_brightness value) \ +{ \ + if (value == LED_OFF) \ + n1200_write(n1200_i2c_client, __reg, N1200_LED_OFF); \ + else \ + n1200_write(n1200_i2c_client, __reg, N1200_LED_ON); \ +} \ +static int n1200_##__name##_blink(struct led_classdev *led_cdev, \ + unsigned long *delay_on, unsigned long *delay_off) \ +{ \ + if (__blinkable == 1) { \ + n1200_write(n1200_i2c_client, __reg, N1200_LED_BLINK); \ + return 0; \ + } else \ + return -EINVAL; \ +} + +N1200_NEW_LED(sata_red, N1200_REG_LED_SATA, 1); +N1200_NEW_LED(system_red, N1200_REG_LED_SYSTEM, 1); +N1200_NEW_LED(busy_orange, N1200_REG_LED_BUSY, 1); + +static int n1200_spkr_event(struct input_dev *dev, unsigned int type, + unsigned int code, int value) +{ + if (type != EV_SND) + return -1; + + switch (code) { + case SND_BELL: + if (value) + value = 1000; + case SND_TONE: + break; + default: + return -1; + } + + if (value > 20 && value < 32767) { + n1200_write(n1200_i2c_client, N1200_REG_BUZZER, N1200_LED_ON); + msleep(value); + n1200_write(n1200_i2c_client, N1200_REG_BUZZER, N1200_LED_OFF); + } else { + n1200_write(n1200_i2c_client, N1200_REG_BUZZER, N1200_LED_OFF); + } + + return 0; +} + +static int n1200_probe(struct i2c_client *client, + const struct i2c_device_id *id) +{ + int version, err = -ENOMEM; + struct mcu_n1200 *mcu; + struct i2c_adapter *adapter = client->adapter; + extern struct machdep_calls ppc_md; + + if (!i2c_check_functionality(client->adapter, I2C_FUNC_SMBUS_BYTE_DATA)) + return -EIO; + + version = n1200_read(client, N1200_REG_VERSION); + if (version < 0) + return -ENODEV; + + mcu = kzalloc(sizeof(struct mcu_n1200), GFP_KERNEL); + if (!mcu) { + dev_err(&client->dev, "failed to allocate driver data\n"); + return -ENOMEM; + } + i2c_set_clientdata(client, mcu); + + mcu->input_dev = input_allocate_device(); + if (!mcu->input_dev) + goto err_alloc; + + dev_info(&adapter->dev, "found mcu_thecus_n1200: v%d\n", version); + + /* Store the client as a static, we'll need it for a poweroff */ + n1200_i2c_client = client; + + mcu->sata_led.name = "n1200:red:sata"; + mcu->sata_led.brightness = LED_OFF; + mcu->sata_led.brightness_set = n1200_sata_red_set; + mcu->sata_led.blink_set = n1200_sata_red_blink; + err = led_classdev_register(&client->dev, &mcu->sata_led); + if (err < 0) { + dev_err(&client->dev, "Failed to register LED sata_red"); + goto err_sata; + } + + mcu->system_led.name = "n1200:red:status"; + mcu->system_led.brightness = LED_OFF; + mcu->system_led.brightness_set = n1200_system_red_set; + mcu->system_led.blink_set = n1200_system_red_blink; + err = led_classdev_register(&client->dev, &mcu->system_led); + if (err < 0) { + dev_err(&client->dev, "Failed to register LED system_red"); + goto err_system; + } + + mcu->busy_led.name = "n1200:orange:status"; + mcu->busy_led.brightness = LED_FULL; + mcu->busy_led.brightness_set = n1200_busy_orange_set; + mcu->busy_led.blink_set = n1200_busy_orange_blink; + err = led_classdev_register(&client->dev, &mcu->busy_led); + if (err < 0) { + dev_err(&client->dev, "Failed to register LED busy_orange"); + goto err_busy; + } + + input_set_drvdata(mcu->input_dev, (void *) mcu); + mcu->input_dev->name = "n1200 beeper", + mcu->input_dev->phys = "n1200/gpio"; + mcu->input_dev->id.bustype = BUS_I2C; + mcu->input_dev->id.vendor = 0x0001; + mcu->input_dev->id.product = 0x0001; + mcu->input_dev->id.version = 0x0100; + mcu->input_dev->dev.parent = &client->dev; + + mcu->input_dev->evbit[0] = BIT_MASK(EV_SND); + mcu->input_dev->sndbit[0] = BIT_MASK(SND_BELL) | BIT_MASK(SND_TONE); + mcu->input_dev->event = n1200_spkr_event; + + err = input_register_device(mcu->input_dev); + if (err) { + dev_err(&client->dev, "Failed to register input device"); + goto err_buzzer; + } + + err = register_reboot_notifier(&n1200_notifier); + if (err != 0) { + dev_err(&client->dev, "Failed to register reboot notifier"); + goto err_notifier; + } + + n1200_write(client, N1200_REG_PWR_STATUS, N1200_PWR_BOOTED); + + ppc_md.power_off = &n1200_power_off; + + return 0; + +err_notifier: + input_unregister_device(mcu->input_dev); + mcu->input_dev = NULL; +err_buzzer: + led_classdev_unregister(&mcu->busy_led); +err_busy: + led_classdev_unregister(&mcu->system_led); +err_system: + led_classdev_unregister(&mcu->sata_led); +err_sata: + input_free_device(mcu->input_dev); +err_alloc: + kfree(mcu); + return err; +} + +static int n1200_remove(struct i2c_client *client) +{ + int err; + extern struct machdep_calls ppc_md; + struct mcu_n1200 *mcu = i2c_get_clientdata(client); + + /* Tell the PIC we are going away, needed if we wanted to reload */ + n1200_write(client, N1200_REG_PWR_STATUS, N1200_PWR_OFF); + + err = unregister_reboot_notifier(&n1200_notifier); + ppc_md.power_off = NULL; + + input_unregister_device(mcu->input_dev); + mcu->input_dev = NULL; + + led_classdev_unregister(&mcu->sata_led); + led_classdev_unregister(&mcu->system_led); + led_classdev_unregister(&mcu->busy_led); + + i2c_set_clientdata(client, NULL); + kfree(mcu); + n1200_i2c_client = NULL; + + return err; +} + +static const struct i2c_device_id n1200_id[] = { + { "mcu_thecus_n1200", 0 }, + { } +}; + +static struct i2c_driver n1200_driver = { + .driver = { + .name = "mcu_thecus_n1200", + }, + .probe = n1200_probe, + .remove = n1200_remove, + .id_table = n1200_id, +}; + +static int __init n1200_init(void) +{ + return i2c_add_driver(&n1200_driver); +} + +static void __exit n1200_exit(void) +{ + i2c_del_driver(&n1200_driver); +} + +MODULE_AUTHOR("Byron Bradley <byron.bbrad...@gmail.com>"); +MODULE_DESCRIPTION("Thecus N1200 Board Controller"); +MODULE_LICENSE("GPL"); + +module_init(n1200_init); +module_exit(n1200_exit); diff --git a/arch/powerpc/platforms/Kconfig b/arch/powerpc/platforms/Kconfig index e3e8707..699ee83 100644 --- a/arch/powerpc/platforms/Kconfig +++ b/arch/powerpc/platforms/Kconfig @@ -329,4 +329,13 @@ config MCU_MPC8349EMITX also register MCU GPIOs with the generic GPIO API, so you'll able to use MCU pins as GPIOs. +config MCU_THECUS_N1200 + tristate "Thecus N1200 MCU driver" + depends on I2C && THECUS_N1200 + select GENERIC_GPIO + select ARCH_REQUIRE_GPIOLIB + help + Say Y here to enable power-off functionality on Thecus N1200 board. + Also provides control over some of the LEDs and the buzzer. + endmenu -- 1.6.3 _______________________________________________ Linuxppc-dev mailing list Linuxppc-dev@ozlabs.org https://ozlabs.org/mailman/listinfo/linuxppc-dev