ds1620.c
上传用户:lgb322
上传日期:2013-02-24
资源大小:30529k
文件大小:8k
源码类别:

嵌入式Linux

开发平台:

Unix_Linux

  1. /*
  2.  * linux/drivers/char/ds1620.c: Dallas Semiconductors DS1620
  3.  *   thermometer driver (as used in the Rebel.com NetWinder)
  4.  */
  5. #include <linux/config.h>
  6. #include <linux/module.h>
  7. #include <linux/sched.h>
  8. #include <linux/miscdevice.h>
  9. #include <linux/smp_lock.h>
  10. #include <linux/delay.h>
  11. #include <linux/proc_fs.h>
  12. #include <linux/capability.h>
  13. #include <linux/init.h>
  14. #include <asm/hardware.h>
  15. #include <asm/mach-types.h>
  16. #include <asm/uaccess.h>
  17. #include <asm/therm.h>
  18. #ifdef CONFIG_PROC_FS
  19. /* define for /proc interface */
  20. #define THERM_USE_PROC
  21. #endif
  22. /* Definitions for DS1620 chip */
  23. #define THERM_START_CONVERT 0xee
  24. #define THERM_RESET 0xaf
  25. #define THERM_READ_CONFIG 0xac
  26. #define THERM_READ_TEMP 0xaa
  27. #define THERM_READ_TL 0xa2
  28. #define THERM_READ_TH 0xa1
  29. #define THERM_WRITE_CONFIG 0x0c
  30. #define THERM_WRITE_TL 0x02
  31. #define THERM_WRITE_TH 0x01
  32. #define CFG_CPU 2
  33. #define CFG_1SHOT 1
  34. static const char *fan_state[] = { "off", "on", "on (hardwired)" };
  35. /*
  36.  * Start of NetWinder specifics
  37.  *  Note!  We have to hold the gpio lock with IRQs disabled over the
  38.  *  whole of our transaction to the Dallas chip, since there is a
  39.  *  chance that the WaveArtist driver could touch these bits to
  40.  *  enable or disable the speaker.
  41.  */
  42. extern spinlock_t gpio_lock;
  43. extern unsigned int system_rev;
  44. static inline void netwinder_ds1620_set_clk(int clk)
  45. {
  46. gpio_modify_op(GPIO_DSCLK, clk ? GPIO_DSCLK : 0);
  47. }
  48. static inline void netwinder_ds1620_set_data(int dat)
  49. {
  50. gpio_modify_op(GPIO_DATA, dat ? GPIO_DATA : 0);
  51. }
  52. static inline int netwinder_ds1620_get_data(void)
  53. {
  54. return gpio_read() & GPIO_DATA;
  55. }
  56. static inline void netwinder_ds1620_set_data_dir(int dir)
  57. {
  58. gpio_modify_io(GPIO_DATA, dir ? GPIO_DATA : 0);
  59. }
  60. static inline void netwinder_ds1620_reset(void)
  61. {
  62. cpld_modify(CPLD_DS_ENABLE, 0);
  63. cpld_modify(CPLD_DS_ENABLE, CPLD_DS_ENABLE);
  64. }
  65. static inline void netwinder_lock(unsigned long *flags)
  66. {
  67. spin_lock_irqsave(&gpio_lock, *flags);
  68. }
  69. static inline void netwinder_unlock(unsigned long *flags)
  70. {
  71. spin_unlock_irqrestore(&gpio_lock, *flags);
  72. }
  73. static inline void netwinder_set_fan(int i)
  74. {
  75. unsigned long flags;
  76. spin_lock_irqsave(&gpio_lock, flags);
  77. gpio_modify_op(GPIO_FAN, i ? GPIO_FAN : 0);
  78. spin_unlock_irqrestore(&gpio_lock, flags);
  79. }
  80. static inline int netwinder_get_fan(void)
  81. {
  82. if ((system_rev & 0xf000) == 0x4000)
  83. return FAN_ALWAYS_ON;
  84. return (gpio_read() & GPIO_FAN) ? FAN_ON : FAN_OFF;
  85. }
  86. /*
  87.  * End of NetWinder specifics
  88.  */
  89. static void ds1620_send_bits(int nr, int value)
  90. {
  91. int i;
  92. for (i = 0; i < nr; i++) {
  93. netwinder_ds1620_set_data(value & 1);
  94. netwinder_ds1620_set_clk(0);
  95. udelay(1);
  96. netwinder_ds1620_set_clk(1);
  97. udelay(1);
  98. value >>= 1;
  99. }
  100. }
  101. static unsigned int ds1620_recv_bits(int nr)
  102. {
  103. unsigned int value = 0, mask = 1;
  104. int i;
  105. netwinder_ds1620_set_data(0);
  106. for (i = 0; i < nr; i++) {
  107. netwinder_ds1620_set_clk(0);
  108. udelay(1);
  109. if (netwinder_ds1620_get_data())
  110. value |= mask;
  111. mask <<= 1;
  112. netwinder_ds1620_set_clk(1);
  113. udelay(1);
  114. }
  115. return value;
  116. }
  117. static void ds1620_out(int cmd, int bits, int value)
  118. {
  119. unsigned long flags;
  120. netwinder_lock(&flags);
  121. netwinder_ds1620_set_clk(1);
  122. netwinder_ds1620_set_data_dir(0);
  123. netwinder_ds1620_reset();
  124. udelay(1);
  125. ds1620_send_bits(8, cmd);
  126. if (bits)
  127. ds1620_send_bits(bits, value);
  128. udelay(1);
  129. netwinder_ds1620_reset();
  130. netwinder_unlock(&flags);
  131. set_current_state(TASK_INTERRUPTIBLE);
  132. schedule_timeout(2);
  133. }
  134. static unsigned int ds1620_in(int cmd, int bits)
  135. {
  136. unsigned long flags;
  137. unsigned int value;
  138. netwinder_lock(&flags);
  139. netwinder_ds1620_set_clk(1);
  140. netwinder_ds1620_set_data_dir(0);
  141. netwinder_ds1620_reset();
  142. udelay(1);
  143. ds1620_send_bits(8, cmd);
  144. netwinder_ds1620_set_data_dir(1);
  145. value = ds1620_recv_bits(bits);
  146. netwinder_ds1620_reset();
  147. netwinder_unlock(&flags);
  148. return value;
  149. }
  150. static int cvt_9_to_int(unsigned int val)
  151. {
  152. if (val & 0x100)
  153. val |= 0xfffffe00;
  154. return val;
  155. }
  156. static void ds1620_write_state(struct therm *therm)
  157. {
  158. ds1620_out(THERM_WRITE_CONFIG, 8, CFG_CPU);
  159. ds1620_out(THERM_WRITE_TL, 9, therm->lo);
  160. ds1620_out(THERM_WRITE_TH, 9, therm->hi);
  161. ds1620_out(THERM_START_CONVERT, 0, 0);
  162. }
  163. static void ds1620_read_state(struct therm *therm)
  164. {
  165. therm->lo = cvt_9_to_int(ds1620_in(THERM_READ_TL, 9));
  166. therm->hi = cvt_9_to_int(ds1620_in(THERM_READ_TH, 9));
  167. }
  168. static ssize_t
  169. ds1620_read(struct file *file, char *buf, size_t count, loff_t *ptr)
  170. {
  171. signed int cur_temp;
  172. signed char cur_temp_degF;
  173. /* Can't seek (pread) on this device */
  174. if (ptr != &file->f_pos)
  175. return -ESPIPE;
  176. cur_temp = cvt_9_to_int(ds1620_in(THERM_READ_TEMP, 9)) >> 1;
  177. /* convert to Fahrenheit, as per wdt.c */
  178. cur_temp_degF = (cur_temp * 9) / 5 + 32;
  179. if (copy_to_user(buf, &cur_temp_degF, 1))
  180. return -EFAULT;
  181. return 1;
  182. }
  183. static int
  184. ds1620_ioctl(struct inode *inode, struct file *file, unsigned int cmd, unsigned long arg)
  185. {
  186. struct therm therm;
  187. int i;
  188. switch(cmd) {
  189. case CMD_SET_THERMOSTATE:
  190. case CMD_SET_THERMOSTATE2:
  191. if (!capable(CAP_SYS_ADMIN))
  192. return -EPERM;
  193. if (cmd == CMD_SET_THERMOSTATE) {
  194. if (get_user(therm.hi, (int *)arg))
  195. return -EFAULT;
  196. therm.lo = therm.hi - 3;
  197. } else {
  198. if (copy_from_user(&therm, (void *)arg, sizeof(therm)))
  199. return -EFAULT;
  200. }
  201. therm.lo <<= 1;
  202. therm.hi <<= 1;
  203. ds1620_write_state(&therm);
  204. break;
  205. case CMD_GET_THERMOSTATE:
  206. case CMD_GET_THERMOSTATE2:
  207. ds1620_read_state(&therm);
  208. therm.lo >>= 1;
  209. therm.hi >>= 1;
  210. if (cmd == CMD_GET_THERMOSTATE) {
  211. if (put_user(therm.hi, (int *)arg))
  212. return -EFAULT;
  213. } else {
  214. if (copy_to_user((void *)arg, &therm, sizeof(therm)))
  215. return -EFAULT;
  216. }
  217. break;
  218. case CMD_GET_TEMPERATURE:
  219. case CMD_GET_TEMPERATURE2:
  220. i = cvt_9_to_int(ds1620_in(THERM_READ_TEMP, 9));
  221. if (cmd == CMD_GET_TEMPERATURE)
  222. i >>= 1;
  223. return put_user(i, (int *)arg) ? -EFAULT : 0;
  224. case CMD_GET_STATUS:
  225. i = ds1620_in(THERM_READ_CONFIG, 8) & 0xe3;
  226. return put_user(i, (int *)arg) ? -EFAULT : 0;
  227. case CMD_GET_FAN:
  228. i = netwinder_get_fan();
  229. return put_user(i, (int *)arg) ? -EFAULT : 0;
  230. case CMD_SET_FAN:
  231. if (!capable(CAP_SYS_ADMIN))
  232. return -EPERM;
  233. if (get_user(i, (int *)arg))
  234. return -EFAULT;
  235. netwinder_set_fan(i);
  236. break;
  237. default:
  238. return -ENOIOCTLCMD;
  239. }
  240. return 0;
  241. }
  242. #ifdef THERM_USE_PROC
  243. static int
  244. proc_therm_ds1620_read(char *buf, char **start, off_t offset,
  245.        int len, int *eof, void *unused)
  246. {
  247. struct therm th;
  248. int temp;
  249. ds1620_read_state(&th);
  250. temp =  cvt_9_to_int(ds1620_in(THERM_READ_TEMP, 9));
  251. len = sprintf(buf, "Thermostat: HI %i.%i, LOW %i.%i; "
  252.       "temperature: %i.%i C, fan %sn",
  253.       th.hi >> 1, th.hi & 1 ? 5 : 0,
  254.       th.lo >> 1, th.lo & 1 ? 5 : 0,
  255.       temp  >> 1, temp  & 1 ? 5 : 0,
  256.       fan_state[netwinder_get_fan()]);
  257. return len;
  258. }
  259. static struct proc_dir_entry *proc_therm_ds1620;
  260. #endif
  261. static struct file_operations ds1620_fops = {
  262. owner: THIS_MODULE,
  263. read: ds1620_read,
  264. ioctl: ds1620_ioctl,
  265. };
  266. static struct miscdevice ds1620_miscdev = {
  267. TEMP_MINOR,
  268. "temp",
  269. &ds1620_fops
  270. };
  271. int __init ds1620_init(void)
  272. {
  273. int ret;
  274. struct therm th, th_start;
  275. if (!machine_is_netwinder())
  276. return -ENODEV;
  277. ds1620_out(THERM_RESET, 0, 0);
  278. ds1620_out(THERM_WRITE_CONFIG, 8, CFG_CPU);
  279. ds1620_out(THERM_START_CONVERT, 0, 0);
  280. /*
  281.  * Trigger the fan to start by setting
  282.  * temperature high point low.  This kicks
  283.  * the fan into action.
  284.  */
  285. ds1620_read_state(&th);
  286. th_start.lo = 0;
  287. th_start.hi = 1;
  288. ds1620_write_state(&th_start);
  289. set_current_state(TASK_INTERRUPTIBLE);
  290. schedule_timeout(2*HZ);
  291. ds1620_write_state(&th);
  292. ret = misc_register(&ds1620_miscdev);
  293. if (ret < 0)
  294. return ret;
  295. #ifdef THERM_USE_PROC
  296. proc_therm_ds1620 = create_proc_entry("therm", 0, 0);
  297. if (proc_therm_ds1620)
  298. proc_therm_ds1620->read_proc = proc_therm_ds1620_read;
  299. else
  300. printk(KERN_ERR "therm: unable to register /proc/thermn");
  301. #endif
  302. ds1620_read_state(&th);
  303. ret = cvt_9_to_int(ds1620_in(THERM_READ_TEMP, 9));
  304. printk(KERN_INFO "Thermostat: high %i.%i, low %i.%i, "
  305.        "current %i.%i C, fan %s.n",
  306.        th.hi >> 1, th.hi & 1 ? 5 : 0,
  307.        th.lo >> 1, th.lo & 1 ? 5 : 0,
  308.        ret   >> 1, ret   & 1 ? 5 : 0,
  309.        fan_state[netwinder_get_fan()]);
  310. return 0;
  311. }
  312. void __exit ds1620_exit(void)
  313. {
  314. #ifdef THERM_USE_PROC
  315. remove_proc_entry("therm", NULL);
  316. #endif
  317. misc_deregister(&ds1620_miscdev);
  318. }
  319. module_init(ds1620_init);
  320. module_exit(ds1620_exit);
  321. MODULE_LICENSE("GPL");