capture_protocol.c 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548
  1. /*
  2. * Copyright (c) 2017 Actions Semiconductor Co., Ltd
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. /**
  7. * @file
  8. * @brief TP Keyboard driver for Actions SoC
  9. */
  10. #define ir_protocol_param_init
  11. #define capture_param_init
  12. #include <shell/shell.h>
  13. #include <drivers/input/input_dev.h>
  14. #include <ir_data_hal.h>
  15. #include <ir_protocol_hal.h>
  16. #include <logging/log.h>
  17. #include "IR_context.h"
  18. struct capture_handle cp_handle;
  19. static int capture_acts_rate_check(u32_t code_rate, u32_t carrier_rate)
  20. {
  21. if((code_rate - carrier_rate) * (code_rate - carrier_rate) > 4)
  22. return -1;
  23. return 0;
  24. }
  25. static int capture_acts_lc_length(u32_t code)
  26. {
  27. u32_t lc_num = 0;
  28. u32_t temp_val = 0;
  29. while(code) {
  30. temp_val = code & 1;
  31. code = code >> 1;
  32. if((code & 1) != temp_val)
  33. lc_num++;
  34. }
  35. return lc_num;
  36. }
  37. static int capture_acts_halfcode_compare(uint32_t code, uint32_t cp, u32_t threshold)
  38. {
  39. int temp_val = 0;
  40. u32_t threshold_val = 0;
  41. temp_val = (code & 0x7fff) - (cp & 0x7fff);
  42. threshold_val = (cp&0x7fff)*threshold/100;
  43. if(temp_val * temp_val > threshold_val * threshold_val)
  44. return -EIO;
  45. return 0;
  46. }
  47. static int capture_acts_lead_check(u32_t *data, u32_t bit_length, u32_t code, u32_t threshold)
  48. {
  49. u32_t lc_num, temp_val, lc_length;
  50. u32_t irpro_buf[32];
  51. memset(irpro_buf, 0, 32);
  52. lc_num = 0;
  53. while(code) {
  54. irpro_buf[lc_num] = irpro_buf[lc_num] + 1;
  55. temp_val = code & 1;
  56. code = code >> 1;
  57. if((code & 1) != temp_val)
  58. lc_num++;
  59. }
  60. for(int j = 0; j < lc_num; j++) {
  61. lc_length = irpro_buf[lc_num - 1 - j] * bit_length;
  62. temp_val = data[j] & 0x7fff;
  63. if(j == (lc_num - 1)) {
  64. if(temp_val > (lc_length + threshold*bit_length/100))
  65. return temp_val -lc_length - threshold*bit_length/200;
  66. }
  67. if((temp_val - lc_length) * (temp_val - lc_length) > (threshold*bit_length/100)*(threshold*bit_length/100))
  68. return -2;
  69. }
  70. return 0;
  71. }
  72. static int capture_acts_code_compare(uint32_t code_pre, uint32_t code_post, uint32_t cp_pre, uint32_t cp_post, u32_t threshold)
  73. {
  74. int temp_val = 0;
  75. u32_t threshold_val = 0;
  76. if((code_pre & 0x8000) != (cp_pre & 0x8000))
  77. return -EIO;
  78. if((code_post & 0x8000) != (cp_post & 0x8000))
  79. return -EIO;
  80. temp_val = (code_pre & 0x7fff) - (cp_pre & 0x7fff);
  81. threshold_val = (cp_pre&0x7fff)*threshold/100;
  82. if(temp_val * temp_val > threshold_val * threshold_val)
  83. return -EIO;
  84. threshold_val = (cp_post&0x7fff)*threshold/100;
  85. if(code_post < (cp_post + threshold_val))//belong to a complete message
  86. return 0;
  87. temp_val = (code_post & 0x7fff) - (cp_post & 0x7fff) - (cp_post & 0x7fff)*threshold/200;//post including a next message
  88. return temp_val;
  89. }
  90. static int capture_acts_cv(u32_t *data, u32_t offset, ir_receive_param_t *param, u32_t threshold, u32_t leader_hc, u32_t *decode)
  91. {
  92. int data_pre, data_post;
  93. uint32_t code_off;
  94. int temp_val, temp_data;
  95. uint32_t code0_pre, code0_post, code1_pre, code1_post;
  96. uint32_t tr0_pre, tr0_post, tr1_pre, tr1_post;
  97. int cp_result[4];
  98. bool code1_discriminate = false;
  99. code0_pre = (param->ir_0_code & 0x8000) | ((param->ir_0_code & 0x7f00) >> 8) * param->code_bit_length;
  100. code0_post = (param->ir_0_code & 0x80) << 8 | (param->ir_0_code & 0x7f) * param->code_bit_length;
  101. code1_pre = (param->ir_1_code & 0x8000) | ((param->ir_1_code & 0x7f00) >> 8) * param->code_bit_length;
  102. code1_post = (param->ir_1_code & 0x80) << 8 | (param->ir_1_code & 0x7f) * param->code_bit_length;
  103. if(param->ir_trc_loc) {
  104. tr0_pre = (param->ir_tr0_code & 0x8000) | ((param->ir_tr0_code & 0x7f00) >> 8) * param->code_bit_length;
  105. tr0_post = (param->ir_tr0_code & 0x80) << 8 | (param->ir_tr0_code & 0x7f) * param->code_bit_length;
  106. tr1_pre = (param->ir_tr1_code & 0x8000) | ((param->ir_tr1_code & 0x7f00) >> 8) * param->code_bit_length;
  107. tr1_post = (param->ir_tr1_code & 0x80) << 8 | (param->ir_tr1_code & 0x7f) * param->code_bit_length;
  108. }
  109. code_off = decode[0] = decode[1] = 0;
  110. if(leader_hc) {
  111. offset = offset - 1;
  112. data_pre = (data[offset] & 0x8000) | leader_hc;
  113. } else
  114. data_pre = data[offset];// a complete message including data_pre | data_ post
  115. data_post = data[offset + 1];
  116. for(int j = offset; j < 99; j++) {
  117. code_off++;
  118. if(data[j] == 0)
  119. break;
  120. if(((data[j] & 0x8000) == 0) && ((data[j] & 0x7fff) > param->ir_max_code))
  121. break;
  122. if(code_off > param->ir_dc_length + 1)
  123. return -3;
  124. if(code_off > param->ir_dc_length)
  125. continue;
  126. if(j > offset)
  127. data_post = data[j + 1];
  128. temp_val = capture_acts_code_compare(data_pre, data_post, code0_pre, code0_post, threshold);
  129. if(temp_val > 0) {
  130. cp_result[0] = capture_acts_halfcode_compare((data_post & 0x8000) | temp_val, code0_pre, threshold);
  131. cp_result[1] = capture_acts_halfcode_compare((data_post & 0x8000) | temp_val, code1_pre, threshold);
  132. cp_result[2] = capture_acts_halfcode_compare((data_post & 0x8000) | temp_val, tr0_pre, threshold);
  133. cp_result[3] = capture_acts_halfcode_compare((data_post & 0x8000) | temp_val, tr1_pre, threshold);
  134. if(cp_result[3]&&cp_result[2]&&cp_result[1]&&cp_result[0])
  135. temp_val = -5;
  136. }
  137. if(temp_val != 0) {
  138. temp_data = capture_acts_code_compare(data_pre, data_post, code1_pre, code1_post, threshold);
  139. if(temp_data > 0) {
  140. cp_result[0] = capture_acts_halfcode_compare((data_post & 0x8000) | temp_data, code0_pre, threshold);
  141. cp_result[1] = capture_acts_halfcode_compare((data_post & 0x8000) | temp_data, code1_pre, threshold);
  142. cp_result[2] = capture_acts_halfcode_compare((data_post & 0x8000) | temp_data, tr0_pre, threshold);
  143. cp_result[3] = capture_acts_halfcode_compare((data_post & 0x8000) | temp_data, tr1_pre, threshold);
  144. if(cp_result[3]&&cp_result[2]&&cp_result[1]&&cp_result[0])
  145. temp_data = -5;
  146. }
  147. if(temp_data >= 0) {
  148. temp_val = temp_data;
  149. code1_discriminate = true;
  150. }
  151. }
  152. if(temp_val >= 0) {
  153. if(code1_discriminate) {
  154. if(param->ir_dc_length - code_off > 32)
  155. decode[1] |= 1 << (param->ir_dc_length - code_off - 32);
  156. else
  157. decode[0] |= 1 << (param->ir_dc_length - code_off);
  158. }
  159. code1_discriminate = false;
  160. }
  161. if(temp_val > 0) {//compare status and including next message
  162. data_pre = (data_post & 0x8000) | temp_val;
  163. continue;
  164. } else if(temp_val == 0) {//campare status
  165. j++;
  166. data_pre = data[j + 1];
  167. continue;
  168. }
  169. if(param->ir_trc_loc) {
  170. temp_val = capture_acts_code_compare(data_pre, data_post, tr0_pre, tr0_post, threshold);
  171. if(temp_val > 0) {
  172. cp_result[0] = capture_acts_halfcode_compare((data_post & 0x8000) | temp_val, code0_pre, threshold);
  173. cp_result[1] = capture_acts_halfcode_compare((data_post & 0x8000) | temp_val, code1_pre, threshold);
  174. cp_result[2] = capture_acts_halfcode_compare((data_post & 0x8000) | temp_val, tr0_pre, threshold);
  175. cp_result[3] = capture_acts_halfcode_compare((data_post & 0x8000) | temp_val, tr1_pre, threshold);
  176. if(cp_result[3]&&cp_result[2]&&cp_result[1]&&cp_result[0])
  177. temp_val = -5;
  178. }
  179. if(temp_val != 0) {
  180. temp_data = capture_acts_code_compare(data_pre, data_post, tr1_pre, tr1_post, threshold);
  181. if(temp_data > 0) {
  182. cp_result[0] = capture_acts_halfcode_compare((data_post & 0x8000) | temp_data, code0_pre, threshold);
  183. cp_result[1] = capture_acts_halfcode_compare((data_post & 0x8000) | temp_data, code1_pre, threshold);
  184. cp_result[2] = capture_acts_halfcode_compare((data_post & 0x8000) | temp_data, tr0_pre, threshold);
  185. cp_result[3] = capture_acts_halfcode_compare((data_post & 0x8000) | temp_data, tr1_pre, threshold);
  186. if(cp_result[3]&&cp_result[2]&&cp_result[1]&&cp_result[0])
  187. temp_data = -5;
  188. }
  189. if(temp_data >= 0) {
  190. temp_val = temp_data;
  191. code1_discriminate = true;
  192. }
  193. }
  194. if(temp_val >= 0) {
  195. if(code1_discriminate) {
  196. if(param->ir_dc_length - code_off > 32)
  197. decode[1] |= 1 << (param->ir_dc_length - code_off - 32);
  198. else
  199. decode[0] |= 1 << (param->ir_dc_length - code_off);
  200. }
  201. code1_discriminate = false;
  202. }
  203. if(temp_val > 0) {//compare status and including next message
  204. data_pre = (data_post & 0x8000) | temp_val;
  205. continue;
  206. } else if(temp_val == 0) {//campare status
  207. j++;
  208. data_pre = data[j + 1];
  209. continue;
  210. }
  211. }
  212. return -3;
  213. }
  214. if(decode[1])
  215. printk("decode:0x%x%x\n", decode[1], decode[0]);
  216. else
  217. printk("decode:0x%x\n", decode[0]);
  218. return 0;
  219. }
  220. static int capture_acts_Tf(u32_t *data, u32_t Tf, u32_t lc_bit_length, u32_t threshold)
  221. {
  222. int data_val, temp_val;
  223. int j = 0;
  224. temp_val = 0;
  225. for(j = 0; j < 100; j++) {
  226. data_val = data[j] & 0x7fff;
  227. temp_val += data_val/10;
  228. if(data_val == 0)
  229. break;
  230. if(((data[j] & 0x8000) == 0) && ((data[j] & 0x7fff) > lc_bit_length))
  231. break;
  232. }
  233. for(; j < 100; j++) {
  234. data_val = data[j] & 0x8000;
  235. if(data_val != 0) {
  236. break;
  237. }
  238. data_val = data[j] & 0x7fff;
  239. if(data_val == 0)
  240. temp_val += 3277/3;
  241. else
  242. temp_val += data_val/10;
  243. }
  244. if(temp_val < Tf - Tf*threshold/100)
  245. return -4;
  246. return 0;
  247. }
  248. static u32_t capture_acts_ic(u32_t *decode, ir_receive_param_t *param)
  249. {
  250. u32_t data[2];
  251. data[0] = ((decode[0] >> param->ir_ic_co) | (decode[1] << (32 - param->ir_ic_co)))&param->ir_ic_mask;
  252. data[1] = (decode[0] >> param->ir_ic_ico)&param->ir_ic_mask;
  253. if(data[0] & data[1])
  254. return -5;
  255. return 0;
  256. }
  257. static int capture_acts_discriminate(struct capture_data *data, u32_t *decode)
  258. {
  259. u32_t irpro_param1, irpro_param2, irpro_param3;
  260. int i, result;
  261. u32_t leader_hc;
  262. result = 0;
  263. for(i = 0; i < Protocols_MAX; i++) {
  264. irpro_param3 = ir_protocol_param[i].ir_threshold_val;
  265. /* rate check */
  266. if(ir_protocol_param[i].ir_cr_check_en) {
  267. if(capture_acts_rate_check(ir_protocol_param[i].ir_cr_rate ,
  268. data->carrier_rate)) {
  269. result = -1;
  270. continue;
  271. }
  272. }
  273. /* lead check */
  274. if(ir_protocol_param[i].ir_lc_check_en) {
  275. irpro_param1 = ir_protocol_param[i].ir_lc_bit_length;
  276. irpro_param2 = ir_protocol_param[i].ir_lc_code;
  277. leader_hc = capture_acts_lead_check(data->capture_data, irpro_param1,
  278. irpro_param2, irpro_param3);
  279. if(leader_hc < 0) {
  280. result = -2;
  281. continue;
  282. }
  283. }
  284. /* code check */
  285. if(ir_protocol_param[i].ir_cv_check_en) {
  286. irpro_param1 = capture_acts_lc_length(ir_protocol_param[i].ir_lc_code);
  287. result = capture_acts_cv(data->capture_data, irpro_param1,
  288. &ir_protocol_param[i], irpro_param3, leader_hc, decode);
  289. if(result == -3) {
  290. continue;
  291. }
  292. }
  293. /* Tf check */
  294. if(ir_protocol_param[i].ir_Tf_check_en) {
  295. irpro_param1 = ir_protocol_param[i].ir_Tf_length;
  296. if(capture_acts_Tf(data->capture_data, irpro_param1,
  297. ir_protocol_param[i].ir_max_code, irpro_param3)) {
  298. result = -4;
  299. continue;
  300. }
  301. }
  302. /* ir inverse code check */
  303. if(ir_protocol_param[i].ir_ic_check_en) {
  304. irpro_param1 = capture_acts_lc_length(ir_protocol_param[i].ir_lc_code);
  305. if(capture_acts_ic(decode, &ir_protocol_param[i])) {
  306. result = -5;
  307. continue;
  308. }
  309. }
  310. break;
  311. }
  312. if(i < Protocols_MAX)
  313. return i;
  314. else
  315. return result;
  316. }
  317. static int capture_acts_protocol(int ret)
  318. {
  319. u32_t protocol;
  320. switch(ret) {
  321. case PWM_IR_6122:
  322. protocol = IR_uPD6121;
  323. printk("IR protocol is nec\n");
  324. break;
  325. case PWM_IR_9012:
  326. protocol = IR_9012;
  327. printk("IR protocol is 9012\n");
  328. break;
  329. case PWM_IR_RC6:
  330. protocol = IR_RC6_dat1;
  331. printk("IR protocol is RC6\n");
  332. break;
  333. case PWM_IR_50462:
  334. protocol = IR_50462;
  335. printk("IR protocol is 50462\n");
  336. break;
  337. case PWM_IR_M50560:
  338. protocol = IR_M50560_dat1;
  339. printk("IR protocol is 50560\n");
  340. break;
  341. case PWM_IR_RC5X:
  342. protocol = IR_RC5;
  343. printk("IR protocol is rc5x\n");
  344. break;
  345. case PWM_IR_7461:
  346. protocol = IR_7461;
  347. printk("IR protocol is 7461\n");
  348. break;
  349. case PWM_IR_3004:
  350. protocol = IR_3004;
  351. printk("IR protocol is 3004\n");
  352. break;
  353. case PWM_IR_RCA:
  354. protocol = IR_RCA;
  355. printk("IR protocol is RCA\n");
  356. break;
  357. default:
  358. printk("can not find right protocol: %d\n", ret);
  359. return -ENOTSUP;
  360. }
  361. return protocol;
  362. }
  363. static void capture_data_dump(struct capture_data *data)
  364. {
  365. printk("rate:%dkhz\n", data->carrier_rate);
  366. for(int i = 0;i < 100;i++) {
  367. data->capture_data[i] = (data->capture_data[i] & 0x8000) | ((data->capture_data[i] & 0x7fff)/3);//cycle to us
  368. if(i%10 == 9) {
  369. printk("%d0%d\n", ((data->capture_data[i] & 0x8000) >> 15), (data->capture_data[i] & 0x7fff));
  370. continue;
  371. }
  372. printk("%d0%d ", ((data->capture_data[i] & 0x8000) >> 15), (data->capture_data[i] & 0x7fff));
  373. }
  374. }
  375. void capture_notify_callback(struct device *dev, struct input_value *val)
  376. {
  377. struct capture_handle * handle = &cp_handle;
  378. struct input_value report_val;
  379. uint32_t decode_data[2] = {0x11,0x22};
  380. struct capture_data data;
  381. int result;
  382. data.capture_data = val->ir.protocol.data;
  383. data.carrier_rate = val->ir.protocol.carry_rate;
  384. capture_data_dump(&data);
  385. result = capture_acts_discriminate(&data, decode_data);
  386. result = capture_acts_protocol(result);
  387. report_val.ir.protocol.mode = result;
  388. report_val.ir.protocol.data = decode_data;
  389. handle->capture_notify(NULL, &report_val);
  390. }
  391. struct capture_handle * capture_device_open(capture_notify_cb cb, char * dev_name)
  392. {
  393. struct capture_handle * handle = &cp_handle;
  394. if(handle == NULL)
  395. {
  396. printk("capture device in mem_malloc failed need %d bytes ",(int)sizeof(struct capture_handle));
  397. return NULL;
  398. }
  399. handle->input_dev = (struct device *)device_get_binding(dev_name);
  400. if (!handle->input_dev) {
  401. printk("cannot found capture dev %s\n",dev_name);
  402. return NULL;
  403. }
  404. input_dev_enable(handle->input_dev);
  405. input_dev_register_notify(handle->input_dev, capture_notify_callback);
  406. handle->capture_notify = cb;
  407. return handle;
  408. }
  409. void capture_device_close(void * handle)
  410. {
  411. struct capture_handle * capture = &cp_handle;
  412. input_dev_unregister_notify(capture->input_dev, capture->capture_notify);
  413. input_dev_disable(capture->input_dev);
  414. }
  415. void capture_device_abort_cb(void * handle)
  416. {
  417. struct capture_handle * capture = &cp_handle;
  418. input_dev_unregister_notify(capture->input_dev, capture->capture_notify);
  419. }
  420. void capture_device_enable_cb(void * handle)
  421. {
  422. struct capture_handle * capture = &cp_handle;
  423. input_dev_register_notify(capture->input_dev, capture->capture_notify);
  424. }