timer.c 1.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081
  1. /*
  2. * Common code to keep time when machine suspends.
  3. *
  4. * Copyright 2007 Johannes Berg <johannes@sipsolutions.net>
  5. *
  6. * GPLv2
  7. */
  8. #include <linux/time.h>
  9. #include <linux/sysdev.h>
  10. #include <asm/rtc.h>
  11. static unsigned long suspend_rtc_time;
  12. /*
  13. * Reset the time after a sleep.
  14. */
  15. static int timer_resume(struct sys_device *dev)
  16. {
  17. struct timeval tv;
  18. struct timespec ts;
  19. struct rtc_time cur_rtc_tm;
  20. unsigned long cur_rtc_time, diff;
  21. /* get current RTC time and convert to seconds */
  22. get_rtc_time(&cur_rtc_tm);
  23. cur_rtc_time = mktime(cur_rtc_tm.tm_year + 1900,
  24. cur_rtc_tm.tm_mon + 1,
  25. cur_rtc_tm.tm_mday,
  26. cur_rtc_tm.tm_hour,
  27. cur_rtc_tm.tm_min,
  28. cur_rtc_tm.tm_sec);
  29. diff = cur_rtc_time - suspend_rtc_time;
  30. /* adjust time of day by seconds that elapsed while
  31. * we were suspended */
  32. do_gettimeofday(&tv);
  33. ts.tv_sec = tv.tv_sec + diff;
  34. ts.tv_nsec = tv.tv_usec * NSEC_PER_USEC;
  35. do_settimeofday(&ts);
  36. return 0;
  37. }
  38. static int timer_suspend(struct sys_device *dev, pm_message_t state)
  39. {
  40. struct rtc_time suspend_rtc_tm;
  41. WARN_ON(!ppc_md.get_rtc_time);
  42. get_rtc_time(&suspend_rtc_tm);
  43. suspend_rtc_time = mktime(suspend_rtc_tm.tm_year + 1900,
  44. suspend_rtc_tm.tm_mon + 1,
  45. suspend_rtc_tm.tm_mday,
  46. suspend_rtc_tm.tm_hour,
  47. suspend_rtc_tm.tm_min,
  48. suspend_rtc_tm.tm_sec);
  49. return 0;
  50. }
  51. static struct sysdev_class timer_sysclass = {
  52. .resume = timer_resume,
  53. .suspend = timer_suspend,
  54. set_kset_name("timer"),
  55. };
  56. static struct sys_device device_timer = {
  57. .id = 0,
  58. .cls = &timer_sysclass,
  59. };
  60. static int time_init_device(void)
  61. {
  62. int error = sysdev_class_register(&timer_sysclass);
  63. if (!error)
  64. error = sysdev_register(&device_timer);
  65. return error;
  66. }
  67. device_initcall(time_init_device);