utils.c 2.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148
  1. #include <common.h>
  2. #include <asm/processor.h>
  3. #include <memio.h>
  4. #include <linux/ctype.h>
  5. static __inline__ unsigned long
  6. get_msr(void)
  7. {
  8. unsigned long msr;
  9. asm volatile("mfmsr %0" : "=r" (msr) :);
  10. return msr;
  11. }
  12. static __inline__ void
  13. set_msr(unsigned long msr)
  14. {
  15. asm volatile("mtmsr %0" : : "r" (msr));
  16. }
  17. static __inline__ unsigned long
  18. get_dec(void)
  19. {
  20. unsigned long val;
  21. asm volatile("mfdec %0" : "=r" (val) :);
  22. return val;
  23. }
  24. static __inline__ void
  25. set_dec(unsigned long val)
  26. {
  27. asm volatile("mtdec %0" : : "r" (val));
  28. }
  29. void
  30. enable_interrupts(void)
  31. {
  32. set_msr (get_msr() | MSR_EE);
  33. }
  34. /* returns flag if MSR_EE was set before */
  35. int
  36. disable_interrupts(void)
  37. {
  38. ulong msr;
  39. msr = get_msr();
  40. set_msr (msr & ~MSR_EE);
  41. return ((msr & MSR_EE) != 0);
  42. }
  43. u8 in8(u32 port)
  44. {
  45. return in_byte(port);
  46. }
  47. void out8(u32 port, u8 val)
  48. {
  49. out_byte(port, val);
  50. }
  51. unsigned long in32(u32 port)
  52. {
  53. return in_long(port);
  54. }
  55. unsigned long simple_strtoul(const char *cp,char **endp,unsigned int base)
  56. {
  57. unsigned long result = 0,value;
  58. if (*cp == '0') {
  59. cp++;
  60. if ((*cp == 'x') && isxdigit(cp[1])) {
  61. base = 16;
  62. cp++;
  63. }
  64. if (!base) {
  65. base = 8;
  66. }
  67. }
  68. if (!base) {
  69. base = 10;
  70. }
  71. while (isxdigit(*cp) && (value = isdigit(*cp) ? *cp-'0' : (islower(*cp)
  72. ? toupper(*cp) : *cp)-'A'+10) < base) {
  73. result = result*base + value;
  74. cp++;
  75. }
  76. if (endp)
  77. *endp = (char *)cp;
  78. return result;
  79. }
  80. long simple_strtol(const char *cp,char **endp,unsigned int base)
  81. {
  82. if(*cp=='-')
  83. return -simple_strtoul(cp+1,endp,base);
  84. return simple_strtoul(cp,endp,base);
  85. }
  86. static inline void
  87. soft_restart(unsigned long addr)
  88. {
  89. /* SRR0 has system reset vector, SRR1 has default MSR value */
  90. /* rfi restores MSR from SRR1 and sets the PC to the SRR0 value */
  91. __asm__ __volatile__ ("mtspr 26, %0" :: "r" (addr));
  92. __asm__ __volatile__ ("li 4, (1 << 6)" ::: "r4");
  93. __asm__ __volatile__ ("mtspr 27, 4");
  94. __asm__ __volatile__ ("rfi");
  95. while(1); /* not reached */
  96. }
  97. void
  98. do_reset (void)
  99. {
  100. ulong addr;
  101. /* flush and disable I/D cache */
  102. __asm__ __volatile__ ("mfspr 3, 1008" ::: "r3");
  103. __asm__ __volatile__ ("ori 5, 5, 0xcc00" ::: "r5");
  104. __asm__ __volatile__ ("ori 4, 3, 0xc00" ::: "r4");
  105. __asm__ __volatile__ ("andc 5, 3, 5" ::: "r5");
  106. __asm__ __volatile__ ("sync");
  107. __asm__ __volatile__ ("mtspr 1008, 4");
  108. __asm__ __volatile__ ("isync");
  109. __asm__ __volatile__ ("sync");
  110. __asm__ __volatile__ ("mtspr 1008, 5");
  111. __asm__ __volatile__ ("isync");
  112. __asm__ __volatile__ ("sync");
  113. #ifdef CFG_RESET_ADDRESS
  114. addr = CFG_RESET_ADDRESS;
  115. #else
  116. /*
  117. * note: when CFG_MONITOR_BASE points to a RAM address,
  118. * CFG_MONITOR_BASE - sizeof (ulong) is usually a valid
  119. * address. Better pick an address known to be invalid on your
  120. * system and assign it to CFG_RESET_ADDRESS.
  121. */
  122. addr = CFG_MONITOR_BASE - sizeof (ulong);
  123. #endif
  124. soft_restart(addr);
  125. while(1); /* not reached */
  126. }