devrtc.c 7.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472
  1. /*
  2. * This file is part of the UCB release of Plan 9. It is subject to the license
  3. * terms in the LICENSE file found in the top-level directory of this
  4. * distribution and at http://akaros.cs.berkeley.edu/files/Plan9License. No
  5. * part of the UCB release of Plan 9, including this file, may be copied,
  6. * modified, propagated, or distributed except according to the terms contained
  7. * in the LICENSE file.
  8. */
  9. #include "u.h"
  10. #include "../port/lib.h"
  11. #include "mem.h"
  12. #include "dat.h"
  13. #include "fns.h"
  14. #include "../port/error.h"
  15. /*
  16. * real time clock and non-volatile ram
  17. */
  18. enum {
  19. Paddr= 0x70, /* address port */
  20. Pdata= 0x71, /* data port */
  21. Seconds= 0x00,
  22. Minutes= 0x02,
  23. Hours= 0x04,
  24. Mday= 0x07,
  25. Month= 0x08,
  26. Year= 0x09,
  27. Status= 0x0A,
  28. Nvoff= 128, /* where usable nvram lives */
  29. Nvsize= 256,
  30. Nbcd= 6,
  31. };
  32. typedef struct Rtc Rtc;
  33. struct Rtc
  34. {
  35. int sec;
  36. int min;
  37. int hour;
  38. int mday;
  39. int mon;
  40. int year;
  41. };
  42. enum{
  43. Qdir = 0,
  44. Qrtc,
  45. Qnvram,
  46. };
  47. Dirtab rtcdir[]={
  48. {".", {Qdir, 0, QTDIR}, 0, 0555},
  49. {"nvram", {Qnvram, 0}, Nvsize, 0664},
  50. {"rtc", {Qrtc, 0}, 0, 0664},
  51. };
  52. static uint32_t rtc2sec(Rtc*);
  53. static void sec2rtc(uint32_t, Rtc*);
  54. void
  55. rtcinit(void)
  56. {
  57. if(ioalloc(Paddr, 2, 0, "rtc/nvr") < 0)
  58. panic("rtcinit: ioalloc failed");
  59. }
  60. static Chan*
  61. rtcattach(char* spec)
  62. {
  63. return devattach('r', spec);
  64. }
  65. static Walkqid*
  66. rtcwalk(Chan* c, Chan *nc, char** name, int nname)
  67. {
  68. return devwalk(c, nc, name, nname, rtcdir, nelem(rtcdir), devgen);
  69. }
  70. static int32_t
  71. rtcstat(Chan* c, uint8_t* dp, int32_t n)
  72. {
  73. return devstat(c, dp, n, rtcdir, nelem(rtcdir), devgen);
  74. }
  75. static Chan*
  76. rtcopen(Chan* c, int omode)
  77. {
  78. Proc *up = externup();
  79. omode = openmode(omode);
  80. switch((uint32_t)c->qid.path){
  81. case Qrtc:
  82. if(strcmp(up->user, eve)!=0 && omode!=OREAD)
  83. error(Eperm);
  84. break;
  85. case Qnvram:
  86. if(strcmp(up->user, eve)!=0)
  87. error(Eperm);
  88. }
  89. return devopen(c, omode, rtcdir, nelem(rtcdir), devgen);
  90. }
  91. static void
  92. rtcclose(Chan* c)
  93. {
  94. }
  95. #define GETBCD(o) ((bcdclock[o]&0xf) + 10*(bcdclock[o]>>4))
  96. static int32_t
  97. rtcextract(void)
  98. {
  99. uint8_t bcdclock[Nbcd];
  100. Rtc rtc;
  101. int i;
  102. /* don't do the read until the clock is no longer busy */
  103. for(i = 0; i < 10000; i++){
  104. outb(Paddr, Status);
  105. if(inb(Pdata) & 0x80)
  106. continue;
  107. /* read clock values */
  108. outb(Paddr, Seconds); bcdclock[0] = inb(Pdata);
  109. outb(Paddr, Minutes); bcdclock[1] = inb(Pdata);
  110. outb(Paddr, Hours); bcdclock[2] = inb(Pdata);
  111. outb(Paddr, Mday); bcdclock[3] = inb(Pdata);
  112. outb(Paddr, Month); bcdclock[4] = inb(Pdata);
  113. outb(Paddr, Year); bcdclock[5] = inb(Pdata);
  114. outb(Paddr, Status);
  115. if((inb(Pdata) & 0x80) == 0)
  116. break;
  117. }
  118. /*
  119. * convert from BCD
  120. */
  121. rtc.sec = GETBCD(0);
  122. rtc.min = GETBCD(1);
  123. rtc.hour = GETBCD(2);
  124. rtc.mday = GETBCD(3);
  125. rtc.mon = GETBCD(4);
  126. rtc.year = GETBCD(5);
  127. /*
  128. * the world starts jan 1 1970
  129. */
  130. if(rtc.year < 70)
  131. rtc.year += 2000;
  132. else
  133. rtc.year += 1900;
  134. return rtc2sec(&rtc);
  135. }
  136. static Lock nvrtlock;
  137. int32_t
  138. rtctime(void)
  139. {
  140. int i;
  141. int32_t t, ot;
  142. ilock(&nvrtlock);
  143. /* loop till we get two reads in a row the same */
  144. t = rtcextract();
  145. for(i = 0; i < 100; i++){
  146. ot = rtcextract();
  147. if(ot == t)
  148. break;
  149. }
  150. iunlock(&nvrtlock);
  151. if(i == 100) print("we are boofheads\n");
  152. return t;
  153. }
  154. static int32_t
  155. rtcread(Chan* c, void* buf, int32_t n, int64_t off)
  156. {
  157. Proc *up = externup();
  158. uint32_t t;
  159. char *a, *start;
  160. uint32_t offset = off;
  161. if(c->qid.type & QTDIR)
  162. return devdirread(c, buf, n, rtcdir, nelem(rtcdir), devgen);
  163. switch((uint32_t)c->qid.path){
  164. case Qrtc:
  165. t = rtctime();
  166. n = readnum(offset, buf, n, t, 12);
  167. return n;
  168. case Qnvram:
  169. if(n == 0)
  170. return 0;
  171. if(n > Nvsize)
  172. n = Nvsize;
  173. a = start = smalloc(n);
  174. ilock(&nvrtlock);
  175. for(t = offset; t < offset + n; t++){
  176. if(t >= Nvsize)
  177. break;
  178. outb(Paddr, Nvoff+t);
  179. *a++ = inb(Pdata);
  180. }
  181. iunlock(&nvrtlock);
  182. if(waserror()){
  183. free(start);
  184. nexterror();
  185. }
  186. memmove(buf, start, t - offset);
  187. poperror();
  188. free(start);
  189. return t - offset;
  190. }
  191. error(Ebadarg);
  192. return 0;
  193. }
  194. #define PUTBCD(n,o) bcdclock[o] = (n % 10) | (((n / 10) % 10)<<4)
  195. static int32_t
  196. rtcwrite(Chan* c, void* buf, int32_t n, int64_t off)
  197. {
  198. Proc *up = externup();
  199. int t;
  200. char *a, *start;
  201. Rtc rtc;
  202. uint32_t secs;
  203. uint8_t bcdclock[Nbcd];
  204. char *cp, *ep;
  205. uint32_t offset = off;
  206. if(offset!=0)
  207. error(Ebadarg);
  208. switch((uint32_t)c->qid.path){
  209. case Qrtc:
  210. /*
  211. * read the time
  212. */
  213. cp = ep = buf;
  214. ep += n;
  215. while(cp < ep){
  216. if(*cp>='0' && *cp<='9')
  217. break;
  218. cp++;
  219. }
  220. secs = strtoul(cp, 0, 0);
  221. /*
  222. * convert to bcd
  223. */
  224. sec2rtc(secs, &rtc);
  225. PUTBCD(rtc.sec, 0);
  226. PUTBCD(rtc.min, 1);
  227. PUTBCD(rtc.hour, 2);
  228. PUTBCD(rtc.mday, 3);
  229. PUTBCD(rtc.mon, 4);
  230. PUTBCD(rtc.year, 5);
  231. /*
  232. * write the clock
  233. */
  234. ilock(&nvrtlock);
  235. outb(Paddr, Seconds); outb(Pdata, bcdclock[0]);
  236. outb(Paddr, Minutes); outb(Pdata, bcdclock[1]);
  237. outb(Paddr, Hours); outb(Pdata, bcdclock[2]);
  238. outb(Paddr, Mday); outb(Pdata, bcdclock[3]);
  239. outb(Paddr, Month); outb(Pdata, bcdclock[4]);
  240. outb(Paddr, Year); outb(Pdata, bcdclock[5]);
  241. iunlock(&nvrtlock);
  242. return n;
  243. case Qnvram:
  244. if(n == 0)
  245. return 0;
  246. if(n > Nvsize)
  247. n = Nvsize;
  248. start = a = smalloc(n);
  249. if(waserror()){
  250. free(start);
  251. nexterror();
  252. }
  253. memmove(a, buf, n);
  254. poperror();
  255. ilock(&nvrtlock);
  256. for(t = offset; t < offset + n; t++){
  257. if(t >= Nvsize)
  258. break;
  259. outb(Paddr, Nvoff+t);
  260. outb(Pdata, *a++);
  261. }
  262. iunlock(&nvrtlock);
  263. free(start);
  264. return t - offset;
  265. }
  266. error(Ebadarg);
  267. return 0;
  268. }
  269. Dev rtcdevtab = {
  270. .dc = 'r',
  271. .name = "rtc",
  272. .reset = devreset,
  273. .init = rtcinit,
  274. .shutdown = devshutdown,
  275. .attach = rtcattach,
  276. .walk = rtcwalk,
  277. .stat = rtcstat,
  278. .open = rtcopen,
  279. .create = devcreate,
  280. .close = rtcclose,
  281. .read = rtcread,
  282. .bread = devbread,
  283. .write = rtcwrite,
  284. .bwrite = devbwrite,
  285. .remove = devremove,
  286. .wstat = devwstat,
  287. };
  288. #define SEC2MIN 60L
  289. #define SEC2HOUR (60L*SEC2MIN)
  290. #define SEC2DAY (24L*SEC2HOUR)
  291. /*
  292. * days per month plus days/year
  293. */
  294. static int dmsize[] =
  295. {
  296. 365, 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31
  297. };
  298. static int ldmsize[] =
  299. {
  300. 366, 31, 29, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31
  301. };
  302. /*
  303. * return the days/month for the given year
  304. */
  305. static int*
  306. yrsize(int y)
  307. {
  308. if((y%4) == 0 && ((y%100) != 0 || (y%400) == 0))
  309. return ldmsize;
  310. else
  311. return dmsize;
  312. }
  313. /*
  314. * compute seconds since Jan 1 1970
  315. */
  316. static uint32_t
  317. rtc2sec(Rtc *rtc)
  318. {
  319. uint32_t secs;
  320. int i;
  321. int *d2m;
  322. secs = 0;
  323. /*
  324. * seconds per year
  325. */
  326. for(i = 1970; i < rtc->year; i++){
  327. d2m = yrsize(i);
  328. secs += d2m[0] * SEC2DAY;
  329. }
  330. /*
  331. * seconds per month
  332. */
  333. d2m = yrsize(rtc->year);
  334. for(i = 1; i < rtc->mon; i++)
  335. secs += d2m[i] * SEC2DAY;
  336. secs += (rtc->mday-1) * SEC2DAY;
  337. secs += rtc->hour * SEC2HOUR;
  338. secs += rtc->min * SEC2MIN;
  339. secs += rtc->sec;
  340. return secs;
  341. }
  342. /*
  343. * compute rtc from seconds since Jan 1 1970
  344. */
  345. static void
  346. sec2rtc(uint32_t secs, Rtc *rtc)
  347. {
  348. int d;
  349. int32_t hms, day;
  350. int *d2m;
  351. /*
  352. * break initial number into days
  353. */
  354. hms = secs % SEC2DAY;
  355. day = secs / SEC2DAY;
  356. if(hms < 0) {
  357. hms += SEC2DAY;
  358. day -= 1;
  359. }
  360. /*
  361. * generate hours:minutes:seconds
  362. */
  363. rtc->sec = hms % 60;
  364. d = hms / 60;
  365. rtc->min = d % 60;
  366. d /= 60;
  367. rtc->hour = d;
  368. /*
  369. * year number
  370. */
  371. if(day >= 0)
  372. for(d = 1970; day >= *yrsize(d); d++)
  373. day -= *yrsize(d);
  374. else
  375. for (d = 1970; day < 0; d--)
  376. day += *yrsize(d-1);
  377. rtc->year = d;
  378. /*
  379. * generate month
  380. */
  381. d2m = yrsize(rtc->year);
  382. for(d = 1; day >= d2m[d]; d++)
  383. day -= d2m[d];
  384. rtc->mday = day + 1;
  385. rtc->mon = d;
  386. return;
  387. }
  388. uint8_t
  389. nvramread(int addr)
  390. {
  391. uint8_t data;
  392. ilock(&nvrtlock);
  393. outb(Paddr, addr);
  394. data = inb(Pdata);
  395. iunlock(&nvrtlock);
  396. return data;
  397. }
  398. void
  399. nvramwrite(int addr, uint8_t data)
  400. {
  401. ilock(&nvrtlock);
  402. outb(Paddr, addr);
  403. outb(Pdata, data);
  404. iunlock(&nvrtlock);
  405. }