ls2k-reset.c 1.1 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253
  1. // SPDX-License-Identifier: GPL-2.0
  2. /*
  3. * Copyright (C) 2021, Qing Zhang <[email protected]>
  4. * Loongson-2K1000 reset support
  5. */
  6. #include <linux/of_address.h>
  7. #include <linux/pm.h>
  8. #include <asm/reboot.h>
  9. #define PM1_STS 0x0c /* Power Management 1 Status Register */
  10. #define PM1_CNT 0x14 /* Power Management 1 Control Register */
  11. #define RST_CNT 0x30 /* Reset Control Register */
  12. static void __iomem *base;
  13. static void ls2k_restart(char *command)
  14. {
  15. writel(0x1, base + RST_CNT);
  16. }
  17. static void ls2k_poweroff(void)
  18. {
  19. /* Clear */
  20. writel((readl(base + PM1_STS) & 0xffffffff), base + PM1_STS);
  21. /* Sleep Enable | Soft Off*/
  22. writel(GENMASK(12, 10) | BIT(13), base + PM1_CNT);
  23. }
  24. static int ls2k_reset_init(void)
  25. {
  26. struct device_node *np;
  27. np = of_find_compatible_node(NULL, NULL, "loongson,ls2k-pm");
  28. if (!np) {
  29. pr_info("Failed to get PM node\n");
  30. return -ENODEV;
  31. }
  32. base = of_iomap(np, 0);
  33. of_node_put(np);
  34. if (!base) {
  35. pr_info("Failed to map PM register base address\n");
  36. return -ENOMEM;
  37. }
  38. _machine_restart = ls2k_restart;
  39. pm_power_off = ls2k_poweroff;
  40. return 0;
  41. }
  42. arch_initcall(ls2k_reset_init);