gpio.c 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937
  1. /******************************************************************************
  2. 版权所有:
  3. 文件名称: gpio.c
  4. 文件版本: 01.01
  5. 创建作者: zhaoyang
  6. 创建日期: 2023-12-14
  7. 功能说明: gpio驱动程序。
  8. 其它说明:
  9. 修改记录:
  10. [new]-->applay
  11. B_DO_ADJ --> GPIOD_17 --> gpio113
  12. B_DO_0 --> GPIOD_03 --> gpio99
  13. B_DO_1 --> GPIOD_19 --> gpio115
  14. B_DO_2 --> GPIOD_02 --> gpio98
  15. B_DO_3 --> GPIOD_09 --> gpio105
  16. B_DO_4 --> GPIOD_18 --> gpio114
  17. B_DO_5 --> GPIOD_16 --> gpio112
  18. B_DO_6 --> GPIOD_07 --> gpio103
  19. B_DO_7 --> GPIOD_08 --> gpio104
  20. B_DO_8 --> GPIOD_06 --> gpio102
  21. T536 start ↓↓↓↓↓↓
  22. RUN_LED --> PB7 --> gpio39
  23. ERR_LED --> PF4 --> gpio164
  24. ESAM_PWR --> PL7 --> gpio159
  25. LINE_LOSS_SET --> PM5 --> gpio389
  26. T536 end ↑↑↑↑↑↑
  27. */
  28. /*------------------------------- 头文件 --------------------------------------
  29. */
  30. #include "bspconfig.h"
  31. #include "gpio.h"
  32. #include "head.h"
  33. #include <rtdm/gpio.h>
  34. /*------------------------------- 宏定义 --------------------------------------
  35. */
  36. #if CFG_BSP_DEBUG
  37. #define _GPIO_DEBUG
  38. #endif
  39. /*------------------------------ 全局变量 -------------------------------------
  40. */
  41. typedef enum {
  42. SYS_GPIO = 0,
  43. RTDM_GPIO = 1,
  44. } GPIO_TYPE;
  45. unsigned short kc_pwm0_stu;
  46. unsigned short kc_pwm1_stu;
  47. unsigned short kc_kout0_stu;
  48. unsigned short kc_kout1_stu;
  49. extern uint32_t g_yx_buf[2]; //预留64个遥信
  50. /*------------------------------ 函数声明 -------------------------------------
  51. */
  52. extern unsigned int change_di_ch(unsigned int di);
  53. int gpio_val_fd[DO_NUM];
  54. int gpio_dev_type[DO_NUM] = {SYS_GPIO,}; //0:系统GPIO, 1:RTDM GPIO
  55. const int gpio_val[DO_NUM] = {98, 99, 115, 98, 105, 114, 112, 103, 104, 102};
  56. int gpio_val_fd_other[GPIO_NUM_OTHER];
  57. int gpio_dev_type_other[GPIO_NUM_OTHER] = {SYS_GPIO,};
  58. const int gpio_val_other[GPIO_NUM_OTHER] ={39,164,359,389};
  59. // static int gpio_watchdog_val = 0;
  60. static int gpio_run_led_val = 0;
  61. /*------------------------------ 外部函数 -------------------------------------
  62. */
  63. #if 1
  64. int sysfs_gpio_unexport(int gpio)
  65. {
  66. char path[64];
  67. int fd;
  68. snprintf(path, sizeof(path), "/sys/class/gpio/unexport");
  69. fd = open(path, O_WRONLY);
  70. if (fd < 0) {
  71. perror("Failed to open unexport");
  72. return -1;
  73. }
  74. char buf[16];
  75. snprintf(buf, sizeof(buf), "%d", gpio);
  76. if (write(fd, buf, strlen(buf)) < 0) {
  77. perror("Failed to set GPIO");
  78. close(fd);
  79. return -1;
  80. }
  81. close(fd);
  82. return 0;
  83. }
  84. int sysfs_gpio_export(int gpio) {
  85. char path[64];
  86. int fd;
  87. // 检查是否已导出
  88. snprintf(path, sizeof(path), "/sys/class/gpio/gpio%d", gpio);
  89. if (access(path, F_OK) == 0) {
  90. sysfs_gpio_unexport(gpio); // bugfix: 避免重复导出
  91. msleep(50); //短暂等待让内核处理
  92. }
  93. snprintf(path, sizeof(path), "/sys/class/gpio/export");
  94. fd = open(path, O_WRONLY);
  95. if (fd < 0) {
  96. perror("Failed to open export");
  97. return -1;
  98. }
  99. char buf[16];
  100. snprintf(buf, sizeof(buf), "%d", gpio);
  101. if (write(fd, buf, strlen(buf)) < 0) {
  102. perror("Failed to export GPIO");
  103. close(fd);
  104. return -1;
  105. }
  106. close(fd);
  107. return 0;
  108. }
  109. int sysfs_gpio_set_direction(int gpio, int is_output) {
  110. char path[64];
  111. int fd;
  112. snprintf(path, sizeof(path), "/sys/class/gpio/gpio%d/direction", gpio);
  113. fd = open(path, O_WRONLY);
  114. if (fd < 0) {
  115. perror("Failed to open direction");
  116. return -1;
  117. }
  118. if (write(fd, is_output ? "out" : "in", is_output ? 3 : 2) < 0) {
  119. perror("Failed to set direction");
  120. close(fd);
  121. return -1;
  122. }
  123. close(fd);
  124. return 0;
  125. }
  126. int sysfs_gpio_set_edge_detect(int gpio, int edge)
  127. {
  128. char path[64];
  129. int fd;
  130. snprintf(path, sizeof(path), "/sys/class/gpio/gpio%d/edge", gpio);
  131. fd = open(path, O_WRONLY);
  132. if (fd < 0) {
  133. perror("Failed to open edge");
  134. close(fd);
  135. return -1;
  136. }
  137. if (write(fd, edge, strlen(edge)) < 0) {
  138. perror("Failed to set edge");
  139. close(fd);
  140. return -1;
  141. }
  142. close(fd);
  143. return 0;
  144. }
  145. int sysfs_gpio_set_value(int gpio, int value) {
  146. char path[64];
  147. int fd = -1;
  148. snprintf(path, sizeof(path), "/sys/class/gpio/gpio%d/value", gpio);
  149. fd = open(path, O_WRONLY);
  150. if (fd < 0) {
  151. perror("Failed to open value");
  152. return -1;
  153. }
  154. if (write(fd, value ? "1" : "0", 1) < 0) {
  155. perror("Failed to set value");
  156. close(fd);
  157. return -1;
  158. }
  159. return fd;
  160. }
  161. int sysfs_gpio_init(int gpio, int direction)
  162. {
  163. #define GPIO_LOW 0
  164. int ret;
  165. ret = sysfs_gpio_export(gpio);
  166. if(ret < 0)
  167. {
  168. printf("sysfs_gpio_export error\n");
  169. return -1;
  170. }
  171. ret = sysfs_gpio_set_direction(gpio, direction);
  172. if(ret < 0)
  173. {
  174. printf("sysfs_gpio_set_direction error\n");
  175. }
  176. return sysfs_gpio_set_value(gpio, GPIO_LOW); // 初始化IO为低电平
  177. }
  178. int is_rtdm_device(int fd) {
  179. if(gpio_dev_type[fd] > SYS_GPIO)
  180. return 1;
  181. else
  182. if(gpio_dev_type[fd] == SYS_GPIO)
  183. return 0;
  184. else
  185. return -1;
  186. }
  187. int rt_write(int fd, char *buf, int len) {
  188. int value;
  189. if(len <= 0 || buf == NULL) {
  190. return -1;
  191. }
  192. if (is_rtdm_device(fd) == 1) {
  193. // Xenomai RTDM设备:需要转换为二进制值
  194. value = atoi(buf);
  195. if(value != 0 && value != 1) {
  196. printf("value :%d ERR, Please enter '0' or '1'\n", value);
  197. return -1;
  198. }
  199. return write(fd, &value, sizeof(value));
  200. }
  201. else
  202. {
  203. // 常规系统设备:直接传递字符串
  204. return write(fd, buf, len);
  205. }
  206. }
  207. int rtdm_gpio_init(int gpio, int direction)
  208. {
  209. #define GPIO_LOW 0
  210. char path[64];
  211. int fd = -1, ret;
  212. int value = 1;
  213. snprintf(path, sizeof(path), "/dev/rtdm/3604000.pinctrl/gpio%d", gpio);
  214. fd = open(path, O_RDWR);
  215. if (fd < 0) {
  216. perror("open");
  217. return -1;
  218. }
  219. ret = ioctl(fd, GPIO_RTIOC_DIR_OUT, &value);
  220. if (ret < 0) {
  221. perror("ioctl DIR_OUT");
  222. close(fd);
  223. return -1;
  224. }
  225. rt_write(gpio_val_fd[DO_OUT0], "0", 2); // 初始化IO为低电平
  226. return fd;
  227. }
  228. int gpio_init(void)
  229. {
  230. #define GPIO_OUTPUT 1
  231. int i;
  232. #ifdef NW_AREA_MAIN_2021
  233. for (i = 0; i < DO_NUM; i ++)
  234. {
  235. gpio_val_fd[i] = rtdm_gpio_init(gpio_val[i], GPIO_OUTPUT);
  236. if(gpio_val_fd[i] < 0)
  237. {
  238. gpio_val_fd[i] = sysfs_gpio_init(gpio_val[i], GPIO_OUTPUT);
  239. if(gpio_val_fd[i] < 0)
  240. {
  241. printf("gpio_val_fd[%d] error\n", i);
  242. gpio_val_fd[i] = -1;
  243. gpio_dev_type[i] = -1;
  244. return -1;
  245. }
  246. }
  247. else {
  248. gpio_dev_type[i] = gpio_val_fd[i]; // 保存RTDM设备类型句柄,这里有风险,fd句柄可能为0
  249. }
  250. }
  251. #endif
  252. for (i = 0; i < GPIO_NUM_OTHER; i ++)
  253. {
  254. gpio_val_fd_other[i] = -1; //rtdm_gpio_init(gpio_val_other[i], GPIO_OUTPUT);
  255. if(gpio_val_fd_other[i] < 0)
  256. {
  257. gpio_val_fd_other[i] = sysfs_gpio_init(gpio_val_other[i], GPIO_OUTPUT);
  258. if(gpio_val_fd_other[i] < 0)
  259. {
  260. printf("gpio_val_fd_other[%d] error\n", i);
  261. gpio_val_fd_other[i] = -1;
  262. gpio_dev_type_other[i] = -1;
  263. return -1;
  264. }
  265. }
  266. else {
  267. gpio_dev_type_other[i] = gpio_val_fd_other[i]; // 保存RTDM设备类型句柄,这里有风险,fd句柄可能为0
  268. }
  269. }
  270. msleep(10);// 延时10ms
  271. kc_pwm0_stu = 0;
  272. kc_pwm1_stu = 0;
  273. kc_kout0_stu=0;
  274. kc_kout1_stu=0;
  275. //其他与底层GPIO相关的配置再此处添加 比如 LED WATCHDOG
  276. write(gpio_val_fd_other[RUN_LED], "0", 2);
  277. gpio_run_led_val = 0;
  278. //上电默认切换到外部232串口
  279. //后续根据投退判断是否需要切换到线损的维护口
  280. write(gpio_val_fd_other[LINE_LOSS_SET], "0", 2);
  281. //开启esam供电
  282. write(gpio_val_fd_other[ESAM_PWR], "0", 2);
  283. #ifdef NW_AREA_MAIN_2021
  284. for (i = 0; i < DO_NUM; i ++)
  285. {
  286. write(gpio_val_fd[i], "1", 2);
  287. }
  288. #endif
  289. return 0;
  290. }
  291. int gpio_exit(void)
  292. {
  293. int i;
  294. #ifdef NW_AREA_MAIN_2021
  295. for (i = 0; i < DO_NUM; i ++)
  296. {
  297. if(gpio_val_fd[i] >= 0)
  298. {
  299. close(gpio_val_fd[i]);
  300. gpio_val_fd[i] = -1;
  301. }
  302. sysfs_gpio_unexport(gpio_val[i]);
  303. }
  304. #endif
  305. for (i = 0; i < GPIO_NUM_OTHER; i ++)
  306. {
  307. if(gpio_val_fd_other[i] >= 0)
  308. {
  309. close(gpio_val_fd_other[i]);
  310. gpio_val_fd_other[i] = -1;
  311. }
  312. sysfs_gpio_unexport(gpio_val_other[i]);
  313. }
  314. return 0;
  315. }
  316. #else
  317. /******************************************************************************
  318. 函数名称: gpio_init
  319. 函数版本: 01.01
  320. 创建作者: sunxi
  321. 创建日期: 2008-06-26
  322. 函数说明: gpio初始化。
  323. 参数说明: 无
  324. 返回值: 成功返回0.
  325. 修改记录:
  326. */
  327. int gpio_init(void)
  328. {
  329. int i, j;
  330. int exportfd, directionfd;
  331. char num[10];
  332. char path[50] = "";
  333. //do
  334. exportfd = open("/sys/class/gpio/export", O_WRONLY);
  335. if (exportfd < 0)
  336. {
  337. printf("Cannot open GPIO to export it\n");
  338. return -1;
  339. }
  340. #ifdef NW_AREA_MAIN_2021
  341. for (i = 0; i < DO_NUM; i ++)
  342. {
  343. sprintf(num, "%d", gpio_val[i]);
  344. write(exportfd, num, 4);
  345. gpio_val_fd[i] = -1;
  346. }
  347. #endif
  348. for (i = 0; i < GPIO_NUM_OTHER; i ++)
  349. {
  350. sprintf(num, "%d", gpio_val_other[i]);
  351. write(exportfd, num, 4);
  352. gpio_val_fd_other[i] = -1;
  353. }
  354. close(exportfd);
  355. #ifdef NW_AREA_MAIN_2021
  356. for (i = 0; i < DO_NUM; i ++)
  357. {
  358. sprintf(path, "/sys/class/gpio/gpio%d/value", gpio_val[i]);
  359. gpio_val_fd[i] = open(path, O_RDWR);
  360. if (gpio_val_fd[i] < 0)
  361. {
  362. printf("Cannot open GPIO%d value!\n", gpio_val[i]);
  363. for(j = 0; j < i; j ++)
  364. {
  365. close(gpio_val_fd[j]);
  366. gpio_val_fd[j] = -1;
  367. }
  368. return -4;
  369. }
  370. }
  371. // Update the direction of the GPIO to be an output
  372. for (i = 0; i < DO_NUM; i ++)
  373. {
  374. sprintf(path, "/sys/class/gpio/gpio%d/direction", gpio_val[i]);
  375. directionfd = open(path, O_RDWR);
  376. if (directionfd < 0)
  377. {
  378. printf("Cannot open GPIO%d direction!\n", gpio_val[i]);
  379. return -2;
  380. }
  381. write(directionfd, "out", 4);
  382. write(gpio_val_fd[i], "1", 2);
  383. close(directionfd);
  384. }
  385. #endif
  386. msleep(10);// 延时10ms
  387. for (i = 0; i < GPIO_NUM_OTHER; i ++)
  388. {
  389. sprintf(path, "/sys/class/gpio/gpio%d/value", gpio_val_other[i]);
  390. gpio_val_fd_other[i] = open(path, O_RDWR);
  391. if (gpio_val_fd_other[i] < 0)
  392. {
  393. printf("Cannot open GPIO%d value!\n", gpio_val_other[i]);
  394. for(j = 0; j < i; j ++)
  395. {
  396. close(gpio_val_fd_other[j]);
  397. gpio_val_fd_other[j] = -1;
  398. }
  399. #ifdef NW_AREA_MAIN_2021
  400. for(j = 0; j < DO_NUM; j ++)
  401. {
  402. close(gpio_val_fd[j]);
  403. gpio_val_fd[j] = -1;
  404. }
  405. #endif
  406. return -5;
  407. }
  408. }
  409. for (i = 0; i < GPIO_NUM_OTHER; i ++)
  410. {
  411. sprintf(path, "/sys/class/gpio/gpio%d/direction", gpio_val_other[i]);
  412. directionfd = open(path, O_RDWR);
  413. if (directionfd < 0)
  414. {
  415. printf("Cannot open GPIO%d direction!\n", gpio_val_other[i]);
  416. return -3;
  417. }
  418. write(directionfd, "out", 4);
  419. close(directionfd);
  420. }
  421. kc_pwm0_stu = 0;
  422. kc_pwm1_stu = 0;
  423. kc_kout0_stu=0;
  424. kc_kout1_stu=0;
  425. //其他与底层GPIO相关的配置再此处添加 比如 LED WATCHDOG jack.liu 202009010
  426. // write(gpio_val_fd_other[WATCHDOG], "0", 2);//TODO: 需根据实际硬件重新对应匹配
  427. // gpio_watchdog_val = 0;
  428. write(gpio_val_fd_other[RUN_LED], "0", 2);
  429. gpio_run_led_val = 0;
  430. //上电默认切换到外部232串口
  431. //后续根据投退判断是否需要切换到线损的维护口
  432. write(gpio_val_fd_other[LINE_LOSS_SET], "0", 2);
  433. //首次上电,开出合闸,会导致第六路也开出,需要初始化关闭
  434. // write(gpio_val_fd[DO_OUT1], "1", 2);
  435. //开启esam供电
  436. write(gpio_val_fd_other[ESAM_PWR], "0", 2);
  437. #ifdef NW_AREA_MAIN_2021
  438. for (i = 0; i < DO_NUM; i ++)
  439. {
  440. write(gpio_val_fd[i], "1", 2);
  441. }
  442. #endif
  443. return 0;
  444. }
  445. int gpio_exit(void)
  446. {
  447. int i;
  448. int unexportfd;
  449. char num[10];
  450. unexportfd = open("/sys/class/gpio/unexport", O_WRONLY);
  451. if (unexportfd < 0)
  452. {
  453. printf("Cannot open GPIO to unexport it\n");
  454. return -1;
  455. }
  456. #ifdef NW_AREA_MAIN_2021
  457. for (i = 0; i < DO_NUM; i ++)
  458. {
  459. if(gpio_val_fd[i] >= 0)
  460. {
  461. close(gpio_val_fd[i]);
  462. gpio_val_fd[i] = -1;
  463. }
  464. sprintf(num, "%d", gpio_val[i]);
  465. write(unexportfd, num, 4);
  466. }
  467. #endif
  468. for (i = 0; i < GPIO_NUM_OTHER; i ++)
  469. {
  470. if(gpio_val_fd_other[i] >= 0)
  471. {
  472. close(gpio_val_fd_other[i]);
  473. gpio_val_fd_other[i] = -1;
  474. }
  475. sprintf(num, "%d", gpio_val_other[i]);
  476. write(unexportfd, num, 4);
  477. }
  478. close(unexportfd);
  479. return 0;
  480. }
  481. #endif
  482. /******************************************************************************
  483. 函数名称: gpio_get_kin
  484. 函数版本: 01.01
  485. 创建作者: sunxi
  486. 创建日期: 2008-09-16
  487. 函数说明: 得到开入的状态。
  488. 参数说明: 无
  489. 返回值: 返回开入的状态,每个bit代表一个开入量。
  490. 修改记录:
  491. */
  492. unsigned int gpio_get_kin(unsigned int addr)
  493. {
  494. register unsigned int dw;
  495. unsigned int value[3];
  496. value[0] = *(volatile unsigned int *)addr;
  497. value[1] = *(volatile unsigned int *)addr;
  498. value[2] = *(volatile unsigned int *)addr;
  499. dw = (value[0] & value[1]) | (value[0] & value[2]) | (value[1] & value[2]);
  500. return dw;
  501. }
  502. /******************************************************************************
  503. 函数名称: gpio_get_di
  504. 函数版本: 01.01
  505. 创建作者:
  506. 创建日期: 2014-11-28
  507. 函数说明: 得到开入的状态
  508. 参数说明: 无
  509. 返回值: 返回开入的状态,每个bit代表一个开入量。
  510. 修改记录:
  511. */
  512. unsigned int gpio_get_di(void)
  513. {
  514. return 0;
  515. #ifndef GW_AREA_MAIN_2021
  516. unsigned int dw = 0;
  517. dw = g_yx_buf[0];//这里保存的是共享内存的yx值,156us更新一次
  518. return change_di_ch(~dw);//取反---因为没有短接时,值为1. 所以要取反为0才行
  519. #endif
  520. }
  521. /******************************************************************************
  522. 函数名称: gpio_di_fj
  523. 函数版本: 01.01
  524. 创建作者:
  525. 创建日期: 2014-11-28
  526. 函数说明: 读取开出返校遥信值,控制板遥信为YX17~YX18
  527. YX17:选择, YX18:执行
  528. 参数说明: 无
  529. 返回值: 返回遥信值
  530. 修改记录:
  531. */
  532. #ifdef DO_KOUT_CHECK
  533. unsigned short gpio_di_fj(unsigned char index)//(void)
  534. #else
  535. unsigned short gpio_di_fj(void)
  536. #endif
  537. {
  538. unsigned int di=0;
  539. unsigned short ret=0;
  540. di = gpio_get_di();
  541. #ifdef DO_KOUT_CHECK
  542. //V4-20;
  543. //ret = (unsigned short)((di >>19) & 0x0001);
  544. ret = (unsigned short)((di >>index) & 0x0001);
  545. #else
  546. ret = (unsigned short)((di >> 23) & 0x0001);//dtu
  547. #endif
  548. return ret;
  549. }
  550. static void gpio_kouten_do(int on, unsigned int kouten)
  551. {
  552. // if (kouten > KOUT_EN1)
  553. // return;
  554. // if(on)
  555. // {
  556. // write(gpio_val_fd_other[KOUT_EN0+kouten], "1", 2);
  557. // }
  558. // else
  559. // {
  560. // write(gpio_val_fd_other[KOUT_EN0+kouten], "0", 2);
  561. // }
  562. }
  563. /******************************************************************************
  564. 函数名称: gpio_kout_do
  565. 函数版本: 01.01
  566. 创建作者:
  567. 创建日期: 2014-11-28
  568. 函数说明: 开出
  569. 参数说明: on: 1开出, 0回收
  570. kout: 那一路
  571. 返回值: 无
  572. 修改记录:
  573. */
  574. void gpio_kout_do(int on, unsigned int kout)
  575. {
  576. //if (kout > (DO_NUM-DO_OUT0))
  577. // return;
  578. if(on)
  579. {
  580. if((kout==6) || (kout==7))
  581. {
  582. // gpio_kouten_do(1,KOUT_EN1);
  583. kc_kout1_stu |= (1 << kout);
  584. }
  585. else
  586. {
  587. // gpio_kouten_do(1,KOUT_EN0);
  588. kc_kout0_stu |= (1 << kout);
  589. }
  590. write(gpio_val_fd[kout+DO_OUT0], "0", 2);
  591. }
  592. else
  593. {
  594. write(gpio_val_fd[kout+DO_OUT0], "1", 2);
  595. if((kout==6) || (kout==7))
  596. {
  597. kc_kout1_stu &= ~(1 << kout);
  598. }
  599. else
  600. {
  601. kc_kout0_stu &= ~(1 << kout);
  602. }
  603. if ((kc_pwm1_stu== 0) &&(kc_kout1_stu == 0))
  604. {
  605. // gpio_kouten_do(0,KOUT_EN1);
  606. }
  607. if ((kc_kout0_stu == 0) && (kc_pwm0_stu==0))
  608. {
  609. // gpio_kouten_do(0,KOUT_EN0);
  610. }
  611. }
  612. }
  613. /******************************************************************************
  614. 函数名称: gpio_pwm_do
  615. 函数版本: 01.01
  616. 创建作者:
  617. 创建日期: 2014-11-28
  618. 函数说明: pwm开出,由于存在kc_pwm_stu,只允许在同一控制路径中开出(例如同一中断或同一线程).
  619. 参数说明: on: 1开出, 0回收
  620. 返回值: 无
  621. 修改记录:
  622. */
  623. void gpio_pwm_do(int on, unsigned int kout)
  624. {
  625. // if (kout >= DO_NUM)
  626. // return;
  627. if(on)
  628. {
  629. if((kout==0) || (kout==1))
  630. {
  631. // gpio_kouten_do(1,KOUT_EN1);
  632. kc_pwm1_stu |= (1 << kout);
  633. }
  634. else
  635. {
  636. // gpio_kouten_do(1,KOUT_EN0);
  637. kc_pwm0_stu |= (1 << kout);
  638. }
  639. write(gpio_val_fd[kout], "0", 2);
  640. }
  641. else
  642. {
  643. write(gpio_val_fd[kout], "1", 2);
  644. if((kout==0) || (kout==1))
  645. {
  646. kc_pwm1_stu &= ~(1 << kout);
  647. }
  648. else
  649. {
  650. kc_pwm0_stu &= ~(1 << kout);
  651. }
  652. if ((kc_pwm1_stu== 0) &&(kc_kout1_stu == 0))
  653. {
  654. // gpio_kouten_do(0,KOUT_EN1);
  655. }
  656. if ((kc_kout0_stu == 0) && (kc_pwm0_stu==0))
  657. {
  658. // gpio_kouten_do(0,KOUT_EN0);
  659. }
  660. }
  661. }
  662. /******************************************************************************
  663. 函数名称: gpio_get_wirelessin
  664. 函数版本: 01.01
  665. 创建作者: sunxi
  666. 创建日期: 2008-09-16
  667. 函数说明: 得到无线遥控的状态。
  668. 参数说明: 无
  669. 返回值: 返回开入的状态,每个bit代表一个开入量。
  670. 修改记录:
  671. */
  672. unsigned int gpio_get_wirelessin(void)
  673. {
  674. register unsigned int dw = 0;
  675. return dw;
  676. }
  677. unsigned int gpio_get_version(void)
  678. {
  679. register unsigned int dw = 0;
  680. return dw;
  681. }
  682. unsigned int gpio_get_addr(void)
  683. {
  684. register unsigned int dw = 0;
  685. return dw;
  686. }
  687. /******************************************************************************
  688. 函数名称: gpio_watchdog_reset
  689. 函数版本: 01.01
  690. 创建作者: sunxi
  691. 创建日期: 2008-09-16
  692. 函数说明: 复位看门狗。
  693. 参数说明: 无。
  694. 返回值: 无。
  695. 修改记录:
  696. */
  697. // void gpio_watchdog_reset(void)
  698. // {
  699. // if(gpio_watchdog_val)
  700. // {
  701. // gpio_watchdog_val = 0;
  702. // write(gpio_val_fd_other[WATCHDOG], "0", 2);
  703. // }
  704. // else
  705. // {
  706. // gpio_watchdog_val = 1;
  707. // write(gpio_val_fd_other[WATCHDOG], "1", 2);
  708. // }
  709. // return;
  710. // }
  711. void _led_run_err(void)
  712. {
  713. static bool berr_led = false;
  714. if(rt_err_count())
  715. {
  716. //错误指示灯常亮且运行熄灭
  717. if(!berr_led)
  718. {
  719. berr_led = false;
  720. write(gpio_val_fd_other[ERR_LED], "0", 2);
  721. write(gpio_val_fd_other[RUN_LED], "1", 2);
  722. }
  723. return;
  724. }
  725. if(berr_led)
  726. {
  727. berr_led = false;
  728. write(gpio_val_fd_other[ERR_LED], "1", 2);
  729. }
  730. //运行指示灯,每隔1s改变状态一次
  731. if(!gpio_run_led_val)
  732. {
  733. gpio_run_led_val = 1;
  734. write(gpio_val_fd_other[RUN_LED], "0", 2);
  735. }
  736. else
  737. {
  738. gpio_run_led_val = 0;
  739. write(gpio_val_fd_other[RUN_LED], "1", 2);
  740. }
  741. }
  742. void esam_power_ctrl(int on)
  743. {
  744. if(on)
  745. write(gpio_val_fd_other[ESAM_PWR], "1", 2);
  746. else
  747. write(gpio_val_fd_other[ESAM_PWR], "0", 2);
  748. }
  749. /*------------------------------ 内部函数 -------------------------------------
  750. */
  751. /*------------------------------ 测试函数 -------------------------------------
  752. */
  753. #ifdef _GPIO_DEBUG
  754. #include "ustimer.h"
  755. #include "rt.h"
  756. int gpio_test(void)
  757. {
  758. int ret = 0;
  759. //unsigned int i;
  760. //gpio_kouten_do(1,KOUT_EN1);
  761. #if 1
  762. while(1)
  763. {
  764. sleep(5);
  765. gpio_kout_do(1,DO_OUT2-DO_OUT0);
  766. //gpio_kout_do(1,DO_OUT3-DO_OUT0);
  767. // gpio_pwm_do(1,DO_PWM1);
  768. sleep(5);
  769. gpio_kout_do(0,DO_OUT2-DO_OUT0);
  770. //gpio_kout_do(0,DO_OUT3-DO_OUT0);
  771. // gpio_pwm_do(0,DO_PWM1);
  772. /*
  773. for(i=DO_OUT0; i<DO_NUM; i++)
  774. {
  775. rt_printf("kout number = %d!\n\r", i);
  776. //msleep();
  777. sleep(1);
  778. gpio_kout_do(1,i-DO_OUT0);
  779. sleep(1);
  780. gpio_kout_do(0,i-DO_OUT0);
  781. }
  782. */
  783. }
  784. #endif
  785. return ret;
  786. }
  787. #endif