| /* |
| * Copyright (C) 2009 Michal Simek <monstr@monstr.eu> |
| * Copyright (C) 2009 PetaLogix |
| * |
| * This file is subject to the terms and conditions of the GNU General Public |
| * License. See the file "COPYING" in the main directory of this archive |
| * for more details. |
| */ |
| |
| #include <linux/init.h> |
| #include <linux/of_platform.h> |
| #include <asm/prom.h> |
| |
| /* Trigger specific functions */ |
| #ifdef CONFIG_GPIOLIB |
| |
| #include <linux/of_gpio.h> |
| |
| static int handle; /* reset pin handle */ |
| static unsigned int reset_val; |
| |
| static int of_reset_gpio_handle(void) |
| { |
| int ret; /* variable which stored handle reset gpio pin */ |
| struct device_node *root; /* root node */ |
| struct device_node *gpio; /* gpio node */ |
| struct gpio_chip *gc; |
| u32 flags; |
| const void *gpio_spec; |
| |
| /* find out root node */ |
| root = of_find_node_by_path("/"); |
| |
| /* give me handle for gpio node to be possible allocate pin */ |
| ret = of_parse_phandles_with_args(root, "hard-reset-gpios", |
| "#gpio-cells", 0, &gpio, &gpio_spec); |
| if (ret) { |
| pr_debug("%s: can't parse gpios property\n", __func__); |
| goto err0; |
| } |
| |
| gc = of_node_to_gpiochip(gpio); |
| if (!gc) { |
| pr_debug("%s: gpio controller %s isn't registered\n", |
| root->full_name, gpio->full_name); |
| ret = -ENODEV; |
| goto err1; |
| } |
| |
| ret = gc->of_xlate(gc, root, gpio_spec, &flags); |
| if (ret < 0) |
| goto err1; |
| |
| ret += gc->base; |
| err1: |
| of_node_put(gpio); |
| err0: |
| pr_debug("%s exited with status %d\n", __func__, ret); |
| return ret; |
| } |
| |
| void of_platform_reset_gpio_probe(void) |
| { |
| int ret; |
| handle = of_reset_gpio_handle(); |
| |
| if (!gpio_is_valid(handle)) { |
| printk(KERN_INFO "Skipping unavailable RESET gpio %d (%s)\n", |
| handle, "reset"); |
| } |
| |
| ret = gpio_request(handle, "reset"); |
| if (ret < 0) { |
| printk(KERN_INFO "GPIO pin is already allocated\n"); |
| return; |
| } |
| |
| /* get current setup value */ |
| reset_val = gpio_get_value(handle); |
| /* FIXME maybe worth to perform any action */ |
| pr_debug("Reset: Gpio output state: 0x%x\n", reset_val); |
| |
| /* Setup GPIO as output */ |
| ret = gpio_direction_output(handle, 0); |
| if (ret < 0) |
| goto err; |
| |
| /* Setup output direction */ |
| gpio_set_value(handle, 0); |
| |
| printk(KERN_INFO "RESET: Registered gpio device: %d, current val: %d\n", |
| handle, reset_val); |
| return; |
| err: |
| gpio_free(handle); |
| return; |
| } |
| |
| |
| static void gpio_system_reset(void) |
| { |
| gpio_set_value(handle, 1 - reset_val); |
| } |
| #else |
| #define gpio_system_reset() do {} while (0) |
| void of_platform_reset_gpio_probe(void) |
| { |
| return; |
| } |
| #endif |
| |
| void machine_restart(char *cmd) |
| { |
| printk(KERN_NOTICE "Machine restart...\n"); |
| gpio_system_reset(); |
| dump_stack(); |
| while (1) |
| ; |
| } |
| |
| void machine_shutdown(void) |
| { |
| printk(KERN_NOTICE "Machine shutdown...\n"); |
| while (1) |
| ; |
| } |
| |
| void machine_halt(void) |
| { |
| printk(KERN_NOTICE "Machine halt...\n"); |
| while (1) |
| ; |
| } |
| |
| void machine_power_off(void) |
| { |
| printk(KERN_NOTICE "Machine power off...\n"); |
| while (1) |
| ; |
| } |