sgm832a_acts.c 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465
  1. /*
  2. * Copyright (c) 2024 Wingcool Technology Co., Ltd
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. /**
  7. * @file
  8. * @brief SGM832A Timer driver for Actions SoC
  9. * typec_num: 0 for typec0, input/output; 1 for typec1, input
  10. */
  11. #include <errno.h>
  12. #include <kernel.h>
  13. #include <string.h>
  14. //#include <stdbool.h>
  15. #include <init.h>
  16. #include <irq.h>
  17. #include <drivers/adc.h>
  18. #include <drivers/input/input_dev.h>
  19. #include <sys/util.h>
  20. #include <sys/byteorder.h>
  21. #include <board.h>
  22. #include <soc_pmu.h>
  23. #include <logging/log.h>
  24. #include <device.h>
  25. #include <drivers/gpio.h>
  26. #include <soc.h>
  27. #include <string.h>
  28. #include <drivers/i2c.h>
  29. //#include <board_cfg.h>
  30. #include <drivers/uart.h>
  31. LOG_MODULE_REGISTER(sgm832a, CONFIG_SYS_LOG_INPUT_DEV_LEVEL);
  32. #define sgm832_typec0_slaver_addr (0x8A >> 1)
  33. #define sgm832_typec1_slaver_addr (0x82 >> 1)
  34. //sgm832 Register Address
  35. #define CONFIGURATION_REGISTER 0x00
  36. #define SHUNT_VOLTAGE_REGISTER 0x01
  37. #define BUS_VOLTAGE_REGISTER 0x02
  38. #define POWER_REGISTER 0x03
  39. #define CURRENT_REGISTER 0x04
  40. #define CALIBRATION_REGISTER 0x05
  41. #define MASK_ENABLE_REGISTER 0x06
  42. #define ALERT_LIMIT_REGISTER 0x07
  43. #define MANUFACTURER_ID_REGISTER 0xFE
  44. #define DIE_ID_REGISTER 0xFF
  45. //#ifndef CONFIG_MERGE_WORK_Q
  46. //#define CONFIG_USED_TP_WORK_QUEUE 0
  47. //#endif
  48. #ifdef CONFIG_USED_TP_WORK_QUEUE
  49. #define CONFIG_TIMER_WORK_Q_STACK_SIZE 1280
  50. struct k_work_q timer_drv_q;
  51. K_THREAD_STACK_DEFINE(timer_work_q_stack, CONFIG_TIMER_WORK_Q_STACK_SIZE);
  52. #endif
  53. struct acts_sgm_data {
  54. input_notify_t notify;
  55. const struct device *i2c_dev;
  56. const struct device *gpio_dev;
  57. const struct device *this_dev;
  58. struct gpio_callback key_gpio_cb;
  59. struct k_work init_timer;
  60. bool inited;
  61. #ifdef CONFIG_PM_DEVICE
  62. uint32_t pm_state;
  63. #endif
  64. };
  65. uint16_t sgm_crc[2] __attribute((used)) = {0};
  66. float f_ShuntVoltage[2] = {0.0}, f_BusVoltage[2] = {0.0}, f_Power[2] = {0.0}, f_Current[2] = {0.0};
  67. static struct acts_sgm_data sgm_acts_ddata;
  68. static void sgm832_typec_get_data(const struct device *i2c_dev, uint16_t sgm832_typec_slaver_addr, uint8_t typec_num);
  69. //static void sgm832_typec1_get_data(const struct device *i2c_dev);
  70. extern void uart2_poll_out_ch(int c);
  71. #include <drivers/hrtimer.h>
  72. #if 1
  73. static struct hrtimer g_sgm_ht_read;
  74. static void sgm832_acts_handler(struct k_work *work)
  75. {
  76. static struct acts_sgm_data *power_consumption = &sgm_acts_ddata;
  77. sgm832_typec_get_data(power_consumption->i2c_dev, sgm832_typec0_slaver_addr, 0); //不在ISR中完成,防止中断嵌套
  78. sgm832_typec_get_data(power_consumption->i2c_dev, sgm832_typec1_slaver_addr, 1); //不在ISR中完成,防止中断嵌套
  79. //sgm832_typec1_get_data(power_consumption->i2c_dev); //不在ISR中完成,防止中断嵌套
  80. }
  81. K_WORK_DEFINE(sgm832_acts, sgm832_acts_handler);
  82. static void htimer_fun(struct hrtimer *ttimer, void *expiry_fn_arg)
  83. {
  84. //static int t;
  85. //printk("%d ---htimer--\n", t++);
  86. k_work_submit(&sgm832_acts); //向系统工作队列提交一个工作项,让工作队列的线程将执行该工作
  87. }
  88. static void htimer_read(unsigned int ms)
  89. {
  90. hrtimer_init(&g_sgm_ht_read, htimer_fun, NULL);
  91. hrtimer_start(&g_sgm_ht_read, 1000*ms, 1000*ms);
  92. }
  93. #endif
  94. static void sgm832_typec_get_data(const struct device *i2c_dev, uint16_t sgm832_typec_slaver_addr, uint8_t typec_num)
  95. {
  96. #if 1
  97. //uint8_t i;
  98. //float f_ShuntVoltage = 0.0, f_BusVoltage = 0.0, f_Power = 0.0, f_Current = 0.0;
  99. //uint16_t u16_ShuntVoltage;
  100. static uint8_t write_cmd[10] = {CONFIGURATION_REGISTER, SHUNT_VOLTAGE_REGISTER,
  101. BUS_VOLTAGE_REGISTER, POWER_REGISTER,
  102. CURRENT_REGISTER, CALIBRATION_REGISTER,
  103. MASK_ENABLE_REGISTER, ALERT_LIMIT_REGISTER,
  104. MANUFACTURER_ID_REGISTER, DIE_ID_REGISTER};
  105. static uint16_t read_data[1] = {0};
  106. int ret = 0;
  107. //printk("sgm832 typec%d get data\n", typec_num);
  108. /*
  109. //00H Configuartion Register
  110. ret = i2c_write_read(i2c_dev, sgm832_typec_slaver_addr, write_cmd, 1, read_data, 2);
  111. if (ret != 0)
  112. {
  113. printk("sgm832 typec%d i2c_write_read Configuartion Register ERR\n", typec_num);
  114. }
  115. read_data[0] = ((read_data[0] & 0xFF00) >> 8) + ((read_data[0] & 0x00FF) << 8);
  116. //printk("ConfigReg: 0x%04x\n", read_data[0]);
  117. */
  118. //01H Shunt Voltage Register
  119. ret = i2c_write_read(i2c_dev, sgm832_typec_slaver_addr, &write_cmd[1], 1, read_data, 2);
  120. if (ret == 0)
  121. {
  122. read_data[0] = ((read_data[0] & 0xFF00) >> 8) + ((read_data[0] & 0x00FF) << 8);
  123. if (read_data[0] & 0x8000) {
  124. // Sign-extend for negative ShuntVoltage
  125. read_data[0] = ~read_data[0] + 1;
  126. }
  127. f_ShuntVoltage[typec_num] = read_data[0] * 2.5 * 0.001;
  128. }
  129. else
  130. {
  131. printk("sgm832 typec%d i2c_write_read Shunt_Voltage ERR\n", typec_num);
  132. }
  133. //printk("Shunt_Voltage: %.4f mV\n", f_ShuntVoltage[typec_num]);
  134. //02H Bus Voltage Register
  135. ret = i2c_write_read(i2c_dev, sgm832_typec_slaver_addr, &write_cmd[2], 1, read_data, 2);
  136. if (ret != 0)
  137. {
  138. printk("sgm832 typec%d i2c_write_read Bus Voltage Register ERR\n", typec_num);
  139. }
  140. read_data[0] = ((read_data[0] & 0xFF00) >> 8) + ((read_data[0] & 0x00FF) << 8);
  141. f_BusVoltage[typec_num] = (float)read_data[0] * 1.25 * 0.001;
  142. //printk("Bus_Voltage: %.3f V\n", f_BusVoltage[typec_num]);
  143. //03H Power Register
  144. ret = i2c_write_read(i2c_dev, sgm832_typec_slaver_addr, &write_cmd[3], 1, read_data, 2);
  145. if (ret != 0)
  146. {
  147. printk("sgm832 typec%d i2c_write_read Power Register ERR\n", typec_num);
  148. }
  149. read_data[0] = ((read_data[0] & 0xFF00) >> 8) + ((read_data[0] & 0x00FF) << 8);
  150. //There is a ratio of 25 between the LSB of the power and the Current_LSB
  151. f_Power[typec_num] = (float)read_data[0] * 25 * 0.00009; //W
  152. //printk("Power: %.3f mW\n", f_Power[typec_num]);
  153. //04H Current Register
  154. ret = i2c_write_read(i2c_dev, sgm832_typec_slaver_addr, &write_cmd[4], 1, read_data, 2);
  155. if (ret == 0)
  156. {
  157. read_data[0] = ((read_data[0] & 0xFF00) >> 8) + ((read_data[0] & 0x00FF) << 8);
  158. if (read_data[0] & 0x8000) {
  159. // Sign-extend for negative Current
  160. read_data[0] = ~read_data[0] + 1;
  161. }
  162. //Current_LSB=Max_Expected_Current/(2^15),
  163. //Max_Expected_Current = 3A,
  164. //Current_LSB=3A/(2^15) ≈ 0.00009A
  165. f_Current[typec_num] = read_data[0] * 0.00009;
  166. }
  167. else
  168. {
  169. printk("sgm832 typec%d i2c_write_read Current Register ERR\n", typec_num);
  170. }
  171. //printk("Current: %.3f mA\n", f_Current[typec_num]);
  172. /*
  173. //05H Calibration Register
  174. ret = i2c_write_read(i2c_dev, sgm832_typec_slaver_addr, &write_cmd[5], 1, read_data, 2);
  175. if (ret != 0)
  176. {
  177. printk("sgm832 typec%d i2c_write_read Calibration Register ERR\n", typec_num);
  178. }
  179. read_data[0] = ((read_data[0] & 0xFF00) >> 8) + ((read_data[0] & 0x00FF) << 8);
  180. //printk("Calibration: 0x%04x\n", read_data[0]);
  181. //06H Mask/Enable Register
  182. ret = i2c_write_read(i2c_dev, sgm832_typec_slaver_addr, &write_cmd[6], 1, read_data, 2);
  183. if (ret != 0)
  184. {
  185. printk("sgm832 typec%d i2c_write_read Mask/Enable Register ERR\n", typec_num);
  186. }
  187. read_data[0] = ((read_data[0] & 0xFF00) >> 8) + ((read_data[0] & 0x00FF) << 8);
  188. //printk("Mask/Enable: 0x%04x\n", read_data[0]);
  189. //07H AlertLimit Register
  190. ret = i2c_write_read(i2c_dev, sgm832_typec_slaver_addr, &write_cmd[7], 1, read_data, 2);
  191. if (ret != 0)
  192. {
  193. printk("sgm832 typec%d i2c_write_read AlertLimit Register ERR\n", typec_num);
  194. }
  195. read_data[0] = ((read_data[0] & 0xFF00) >> 8) + ((read_data[0] & 0x00FF) << 8);
  196. //printk("AlertLimit: 0x%04x\n", read_data[0]);
  197. //FEH Manufacturer ID Register
  198. ret = i2c_write_read(i2c_dev, sgm832_typec_slaver_addr, &write_cmd[8], 1, read_data, 2);
  199. if (ret != 0)
  200. {
  201. printk("sgm832 typec%d i2c_write_read Manufacturer ID Register ERR\n", typec_num);
  202. }
  203. read_data[0] = ((read_data[0] & 0xFF00) >> 8) + ((read_data[0] & 0x00FF) << 8);
  204. //printk("Manufacturer_ID: 0x%04x\n", read_data[0]);
  205. //FFH Die ID Register
  206. ret = i2c_write_read(i2c_dev, sgm832_typec_slaver_addr, &write_cmd[9], 1, read_data, 2);
  207. if (ret != 0)
  208. {
  209. printk("sgm832 typec%d i2c_write_read Die ID Register ERR\n", typec_num);
  210. }
  211. read_data[0] = ((read_data[0] & 0xFF00) >> 8) + ((read_data[0] & 0x00FF) << 8);
  212. //printk("Die_ID: 0x%04x\n", read_data[0]);
  213. */
  214. #endif
  215. }
  216. #if 0
  217. static void sgm832_typec1_get_shuntvoltage(const struct device *i2c_dev)
  218. {
  219. #if 1
  220. //uint8_t i;
  221. float f_ShuntVoltage = 0.0, f_BusVoltage = 0.0;
  222. //uint16_t u16_ShuntVoltage;
  223. static uint8_t write_cmd[2] = {CONFIGURATION_REGISTER, MANUFACTURER_ID_REGISTER};
  224. static uint16_t read_data[8] = {0};
  225. int ret = 0;
  226. printk("sgm832 typec1 get data\n");
  227. ret = i2c_write_read(i2c_dev, sgm832_typec1_slaver_addr, write_cmd, 1, read_data, sizeof(read_data));
  228. //ret = i2c_burst_read(i2c_dev, sgm832_typec1_slaver_addr, CONFIGURATION_REGISTER, read_data, sizeof(read_data));
  229. //ret = i2c_read(i2c_dev, read_data, 7, sgm832_typec1_slaver_addr);
  230. if (ret == 0)
  231. {
  232. if (read_data[1] & 0x8000) {
  233. // Sign-extend for negative ShuntVoltage
  234. read_data[1] = ~read_data[1] + 1;
  235. //printf( "ShuntVoltage: 0x%04x\n", u16_ShuntVoltage);
  236. f_ShuntVoltage = read_data[1] * 2.5 * 0.001;
  237. }
  238. else {
  239. f_ShuntVoltage = read_data[1] * 2.5 * 0.001;
  240. }
  241. //read_data[2] = ((read_data[2] & 0xFF00) >> 8) + ((read_data[2] & 0x00FF) << 8);
  242. f_BusVoltage = (float)read_data[2] * 1.25 * 0.001;
  243. //uart2_poll_out_ch(read_data[i]); //uart2 send data
  244. }
  245. else
  246. {
  247. printk("sgm832 typec1 i2c_write_read ERR\n");
  248. }
  249. printk("ConfigReg: 0x%04x, ShuntVoltage: %.4f mV, BusVoltage: %.3f V, Power: 0x%04x, Current: 0x%04x, Calibration: 0x%04x, MaskEnable: 0x%04x, AlertLimit: 0x%04x\n",
  250. read_data[0], f_ShuntVoltage, f_BusVoltage, read_data[3], read_data[4], read_data[5], read_data[6], read_data[7]);
  251. ret = i2c_write_read(i2c_dev, sgm832_typec1_slaver_addr, &write_cmd[1], 1, read_data, 4);
  252. //ret = i2c_burst_read(i2c_dev, sgm832_typec1_slaver_addr, CONFIGURATION_REGISTER, read_data, sizeof(read_data));
  253. //ret = i2c_read(i2c_dev, read_data, 7, sgm832_typec1_slaver_addr);
  254. if (ret == 0)
  255. {
  256. }
  257. else
  258. {
  259. printk("sgm832 typec1 i2c_write_read ERR\n");
  260. }
  261. printk("ManufacturerID: 0x%04x, DieID: 0x%04x\n",
  262. read_data[0], read_data[1]);
  263. #endif
  264. }
  265. #endif
  266. /*
  267. * sgm832_typec_calibration
  268. * @i2c_dev: i2c device, sgm832 typec i2c device,
  269. * @sgm832_typec_slaver_addr: sgm832 typec i2c slaver address,
  270. * @typec_num: typec number, 0 or 1
  271. * @return: none
  272. * @description: Set sgm832 calibration register, CAL=0.00512/(Current_LSB*Rshunt), Current_LSB=Max_Expected_Current/(2^15),
  273. * Max_Expected_Current = 3A, Rshunt = 0.028Ω, Current_LSB=3A/(2^15) ≈ 0.00009A,
  274. * CAL=0.00512/(0.00009A*0.028Ω) ≈ 2000 = 0x07D0, so CAL = 0x07D0
  275. */
  276. static void sgm832_typec_calibration(const struct device *i2c_dev, uint16_t sgm832_typec_slaver_addr, uint8_t typec_num)
  277. {
  278. //static uint8_t configurate_write_data[3] = {CONFIGURATION_REGISTER, 0x47, 0x6F};
  279. static uint8_t calibrate_write_data0[3] = {CALIBRATION_REGISTER, 0x07, 0xcd}; //typec0, input/output
  280. static uint8_t calibrate_write_data1[3] = {CALIBRATION_REGISTER, 0x07, 0xcd}; //typec1, input
  281. int ret = 0;
  282. //ret = i2c_write(i2c_dev, configurate_write_data, 3, sgm832_typec_slaver_addr);
  283. //if (ret != 0)
  284. //{
  285. // printk("i2c write sgm832_typec%d Configuration Register ERR\n", typec_num);
  286. // return;
  287. //}
  288. switch(typec_num)
  289. {
  290. case 0:
  291. ret = i2c_write(i2c_dev, calibrate_write_data0, 3, sgm832_typec_slaver_addr);
  292. break;
  293. case 1:
  294. ret = i2c_write(i2c_dev, calibrate_write_data1, 3, sgm832_typec_slaver_addr);
  295. break;
  296. default:
  297. break;
  298. }
  299. if (ret != 0)
  300. {
  301. printk("i2c write sgm832_typec%d Calibration Register ERR\n", typec_num);
  302. }
  303. }
  304. static void _sgm832a_init_work(struct k_work *work)
  305. {
  306. struct acts_sgm_data *power_consumption = &sgm_acts_ddata;
  307. printk("sgm832a init work\n");
  308. power_consumption->inited = true;
  309. //write Calibration Register
  310. sgm832_typec_calibration(power_consumption->i2c_dev, sgm832_typec0_slaver_addr, 0);
  311. sgm832_typec_calibration(power_consumption->i2c_dev, sgm832_typec1_slaver_addr, 1);
  312. htimer_read(500); //1000ms = 1s
  313. }
  314. static int _sgm832a_acts_init(const struct device *dev)
  315. {
  316. struct acts_sgm_data *power_consumption = dev->data;
  317. printk("sgm832a acts init\n");
  318. #if 1
  319. power_consumption->this_dev = (struct device *)dev;
  320. power_consumption->i2c_dev = (struct device *)device_get_binding(CONFIG_SGM832A_I2C_NAME);
  321. if (!power_consumption->i2c_dev) {
  322. printk("can not access right i2c device\n");
  323. return -1;
  324. }
  325. power_consumption->inited = false;
  326. k_work_init(&power_consumption->init_timer, _sgm832a_init_work);
  327. #ifdef CONFIG_USED_TP_WORK_QUEUE
  328. k_work_queue_start(&timer_drv_q, timer_work_q_stack, K_THREAD_STACK_SIZEOF(timer_work_q_stack), 7, NULL);
  329. k_work_submit_to_queue(&timer_drv_q, &power_consumption->init_timer);
  330. #else
  331. k_work_submit(&power_consumption->init_timer);
  332. #endif
  333. #endif
  334. printk("sgm832a acts init exit\n");
  335. return 0;
  336. }
  337. #ifdef CONFIG_PM_DEVICE
  338. static void _sgm832a_suspend(const struct device *dev)
  339. {
  340. //struct acts_sgm_data *power_consumption = (struct acts_sgm_data *)dev->data;
  341. printk("sgm832a suspend\n");
  342. hrtimer_stop(&g_sgm_ht_read);
  343. }
  344. static void _sgm832a_resume(const struct device *dev)
  345. {
  346. struct acts_sgm_data *power_consumption = (struct acts_sgm_data *)dev->data;
  347. power_consumption->i2c_dev = (struct device *)device_get_binding(CONFIG_SGM832A_I2C_NAME);
  348. if (!power_consumption->i2c_dev) {
  349. printk("can not access right i2c device\n");
  350. return;
  351. }
  352. power_consumption->inited = false;
  353. k_work_init(&power_consumption->init_timer, _sgm832a_init_work);
  354. printk("sgm832a resume\n");
  355. #ifdef CONFIG_USED_TP_WORK_QUEUE
  356. k_work_submit_to_queue(&tp_drv_q, &power_consumption->init_timer);
  357. #else
  358. k_work_submit(&power_consumption->init_timer);
  359. #endif
  360. }
  361. static int _sgm832a_pm_control(const struct device *dev, enum pm_device_action action)
  362. {
  363. int ret = 0;
  364. //printk("sgm832a pm control\n");
  365. switch (action) {
  366. case PM_DEVICE_ACTION_SUSPEND:
  367. break;
  368. case PM_DEVICE_ACTION_RESUME:
  369. break;
  370. case PM_DEVICE_ACTION_EARLY_SUSPEND:
  371. _sgm832a_suspend(dev);
  372. break;
  373. case PM_DEVICE_ACTION_LATE_RESUME:
  374. _sgm832a_resume(dev);
  375. break;
  376. default:
  377. break;
  378. }
  379. return ret;
  380. }
  381. #else /* CONFIG_PM_DEVICE */
  382. static int _sgm832a_pm_control(const struct device *dev, uint32_t ctrl_command,
  383. void *context, device_pm_cb cb, void *arg)
  384. {
  385. }
  386. #endif
  387. #if IS_ENABLED(CONFIG_SGM832A)
  388. DEVICE_DEFINE(sgm832a, CONFIG_SGM832A_DEV_NAME, _sgm832a_acts_init,
  389. _sgm832a_pm_control, &sgm_acts_ddata, NULL, POST_KERNEL,
  390. 50, NULL);
  391. #endif