bsp_share.c 6.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262
  1. #include <sys/types.h>
  2. #include <sys/mman.h>
  3. #include <sys/stat.h>
  4. #include <sys/ioctl.h>
  5. #include <errno.h>
  6. #include <fcntl.h>
  7. #include <unistd.h>
  8. #include <stdio.h>
  9. #include <stdlib.h>
  10. #include <string.h>
  11. #include "bsp_share.h"
  12. struct shmem_fd shmem_fd;
  13. #define SHMEM_WR_FLAG 0x1122
  14. #define SHMEM_RD_FLAG 0x3344
  15. /* CRC-16 表 */
  16. static const unsigned short Lookup[256]= {
  17. 0x0000,0xC0C1,0xC181,0x0140,0xC301,0x03C0,0x0280,0xC241,
  18. 0xC601,0x06C0,0x0780,0xC741,0x0500,0xC5C1,0xC481,0x0440,
  19. 0xCC01,0x0CC0,0x0D80,0xCD41,0x0F00,0xCFC1,0xCE81,0x0E40,
  20. 0x0A00,0xCAC1,0xCB81,0x0B40,0xC901,0x09C0,0x0880,0xC841,
  21. 0xD801,0x18C0,0x1980,0xD941,0x1B00,0xDBC1,0xDA81,0x1A40,
  22. 0x1E00,0xDEC1,0xDF81,0x1F40,0xDD01,0x1DC0,0x1C80,0xDC41,
  23. 0x1400,0xD4C1,0xD581,0x1540,0xD701,0x17C0,0x1680,0xD641,
  24. 0xD201,0x12C0,0x1380,0xD341,0x1100,0xD1C1,0xD081,0x1040,
  25. 0xF001,0x30C0,0x3180,0xF141,0x3300,0xF3C1,0xF281,0x3240,
  26. 0x3600,0xF6C1,0xF781,0x3740,0xF501,0x35C0,0x3480,0xF441,
  27. 0x3C00,0xFCC1,0xFD81,0x3D40,0xFF01,0x3FC0,0x3E80,0xFE41,
  28. 0xFA01,0x3AC0,0x3B80,0xFB41,0x3900,0xF9C1,0xF881,0x3840,
  29. 0x2800,0xE8C1,0xE981,0x2940,0xEB01,0x2BC0,0x2A80,0xEA41,
  30. 0xEE01,0x2EC0,0x2F80,0xEF41,0x2D00,0xEDC1,0xEC81,0x2C40,
  31. 0xE401,0x24C0,0x2580,0xE541,0x2700,0xE7C1,0xE681,0x2640,
  32. 0x2200,0xE2C1,0xE381,0x2340,0xE101,0x21C0,0x2080,0xE041,
  33. 0xA001,0x60C0,0x6180,0xA141,0x6300,0xA3C1,0xA281,0x6240,
  34. 0x6600,0xA6C1,0xA781,0x6740,0xA501,0x65C0,0x6480,0xA441,
  35. 0x6C00,0xACC1,0xAD81,0x6D40,0xAF01,0x6FC0,0x6E80,0xAE41,
  36. 0xAA01,0x6AC0,0x6B80,0xAB41,0x6900,0xA9C1,0xA881,0x6840,
  37. 0x7800,0xB8C1,0xB981,0x7940,0xBB01,0x7BC0,0x7A80,0xBA41,
  38. 0xBE01,0x7EC0,0x7F80,0xBF41,0x7D00,0xBDC1,0xBC81,0x7C40,
  39. 0xB401,0x74C0,0x7580,0xB541,0x7700,0xB7C1,0xB681,0x7640,
  40. 0x7200,0xB2C1,0xB381,0x7340,0xB101,0x71C0,0x7080,0xB041,
  41. 0x5000,0x90C1,0x9181,0x5140,0x9301,0x53C0,0x5280,0x9241,
  42. 0x9601,0x56C0,0x5780,0x9741,0x5500,0x95C1,0x9481,0x5440,
  43. 0x9C01,0x5CC0,0x5D80,0x9D41,0x5F00,0x9FC1,0x9E81,0x5E40,
  44. 0x5A00,0x9AC1,0x9B81,0x5B40,0x9901,0x59C0,0x5880,0x9841,
  45. 0x8801,0x48C0,0x4980,0x8941,0x4B00,0x8BC1,0x8A81,0x4A40,
  46. 0x4E00,0x8EC1,0x8F81,0x4F40,0x8D01,0x4DC0,0x4C80,0x8C41,
  47. 0x4400,0x84C1,0x8581,0x4540,0x8701,0x47C0,0x4680,0x8641,
  48. 0x8201,0x42C0,0x4380,0x8341,0x4100,0x81C1,0x8081,0x4040
  49. };
  50. static unsigned short crc16(unsigned char *p, int len)
  51. {
  52. unsigned short crc = 0;
  53. int zero = 1;
  54. if (len <= 0)
  55. return crc;
  56. while(len-- > 0) {
  57. if(*p != 0)
  58. zero = 0;
  59. crc = Lookup[(crc & 0xff) ^ (*p++)] ^ (crc >> 8);
  60. }
  61. if(zero)
  62. crc=0x5678; /* 若全为0,crc返回就不要为零 */
  63. return crc;
  64. }
  65. int shmem_read_crc(struct shmem_fd *shmem_fd, void *buf, int len,
  66. int waiting, unsigned short *crc)
  67. {
  68. volatile struct shmem_block *linux_read = shmem_fd->linux_read;
  69. unsigned short _crc;
  70. if (!buf)
  71. return -EINVAL;
  72. if (len > shmem_fd->linux_read_data_len) {
  73. printf("len(%d) != MAX_SHMEM_RD_DATA_LEN(%d)\n",
  74. len, shmem_fd->linux_read_data_len);
  75. return -EINVAL;
  76. }
  77. if (linux_read->flag == SHMEM_RD_FLAG)
  78. goto read;
  79. if (waiting < 0) {
  80. while (linux_read->flag != SHMEM_RD_FLAG);
  81. } else if (waiting > 0) {
  82. while (waiting-- && linux_read->flag != SHMEM_RD_FLAG)
  83. usleep(waiting);
  84. }
  85. if (linux_read->flag != SHMEM_RD_FLAG)
  86. return -EBUSY;
  87. read:
  88. _crc = crc16((void *)linux_read->data, linux_read->len);
  89. if (_crc != linux_read->crc) {
  90. printf("crc(%x) != linux_read->crc(%x)\n",
  91. _crc, linux_read->crc);
  92. return -EIO;
  93. }
  94. if (len > linux_read->len)
  95. len = linux_read->len;
  96. memcpy(buf, (const void *)linux_read->data, len);
  97. linux_read->flag = 0;
  98. *crc = _crc;
  99. return len;
  100. }
  101. int shmem_write_crc(struct shmem_fd *shmem_fd, void *buf, int len,
  102. int waiting, unsigned short *crc)
  103. {
  104. volatile struct shmem_block *linux_write = shmem_fd->linux_write;
  105. unsigned short _crc;
  106. if (!buf)
  107. return -EINVAL;
  108. if (len > shmem_fd->linux_write_data_len) {
  109. printf("len(%d) != MAX_SHMEM_WR_DATA_LEN(%d)\n",
  110. len, shmem_fd->linux_write_data_len);
  111. return -EINVAL;
  112. }
  113. if (linux_write->flag)
  114. goto write;
  115. if (waiting < 0) {
  116. while (linux_write->flag);
  117. } else if (waiting > 0) {
  118. while (waiting-- && linux_write->flag)
  119. usleep(waiting);
  120. }
  121. if (linux_write->flag == SHMEM_WR_FLAG)
  122. return -EBUSY;
  123. write:
  124. memcpy((void *)linux_write->data, buf, len);
  125. _crc = crc16((void *)linux_write->data, len);
  126. linux_write->crc = _crc;
  127. linux_write->len = len;
  128. linux_write->flag = SHMEM_WR_FLAG;
  129. *crc = _crc;
  130. return len;
  131. }
  132. int shmem_read(struct shmem_fd *shmem_fd, void *buf, int len, int waiting)
  133. {
  134. volatile struct shmem_block *linux_read = shmem_fd->linux_read;
  135. if (!buf || len > shmem_fd->linux_read_data_len)
  136. return -1;
  137. if (linux_read->flag == SHMEM_RD_FLAG)
  138. goto read;
  139. if (waiting < 0) {
  140. while (linux_read->flag != SHMEM_RD_FLAG);
  141. } else if (waiting > 0) {
  142. while (waiting-- && linux_read->flag != SHMEM_RD_FLAG)
  143. usleep(waiting);
  144. }
  145. if (linux_read->flag != SHMEM_RD_FLAG)
  146. return -EBUSY;
  147. read:
  148. if (len > linux_read->len)
  149. len = linux_read->len;
  150. memcpy(buf, (const void *)linux_read->data, len);
  151. linux_read->flag = 0;
  152. return 0;
  153. }
  154. int shmem_write(struct shmem_fd *shmem_fd, void *buf, int len, int waiting)
  155. {
  156. volatile struct shmem_block *linux_write = shmem_fd->linux_write;
  157. if (!buf || len > shmem_fd->linux_write_data_len)
  158. return -1;
  159. if (linux_write->flag)
  160. goto write;
  161. if (waiting < 0) {
  162. while (linux_write->flag);
  163. } else if (waiting > 0) {
  164. while (waiting-- && linux_write->flag)
  165. usleep(waiting);
  166. }
  167. if (linux_write->flag == SHMEM_WR_FLAG)
  168. return -EBUSY;
  169. write:
  170. memcpy((void *)linux_write->data, buf, len);
  171. linux_write->len = len;
  172. linux_write->flag = SHMEM_WR_FLAG;
  173. return 0;
  174. }
  175. #define ARV_SHMEM_REGION_LEN 0x10
  176. int shmem_init(struct shmem_fd *shmem_fd)
  177. {
  178. int fd;
  179. void *shmem;
  180. int ret;
  181. fd = open("/dev/arv-shmem", O_RDWR);
  182. if (fd == -1) {
  183. printf("can not open file /dev/arv-shmem\n");
  184. return -1;
  185. }
  186. ret = ioctl(fd, ARV_SHMEM_REGION_LEN, &shmem_fd->shmem_len);
  187. if (ret) {
  188. printf("failed to get the len of shmem\n");
  189. return -1;
  190. }
  191. printf("shmem len: 0x%x\n", shmem_fd->shmem_len);
  192. shmem_fd->linux_write_len = max_shmem_wr_len(shmem_fd->shmem_len);
  193. shmem_fd->linux_read_len = max_shmem_rd_len(shmem_fd->shmem_len);
  194. shmem_fd->linux_write_data_len = max_shmem_wr_data_len(shmem_fd->shmem_len);
  195. shmem_fd->linux_read_data_len = max_shmem_rd_data_len(shmem_fd->shmem_len);
  196. shmem = mmap(NULL, shmem_fd->shmem_len, PROT_READ|PROT_WRITE,
  197. MAP_SHARED, fd, 0);
  198. if (shmem <= 0) {
  199. printf("failed to get shmem\n");
  200. return -1;
  201. }
  202. printf("mmap: %llx\n", (unsigned long long)shmem);
  203. shmem_fd->fd = fd;
  204. shmem_fd->linux_write = (struct shmem_block *)shmem;
  205. shmem_fd->linux_read = (struct shmem_block *)(shmem + shmem_fd->linux_write_len);
  206. return 0;
  207. }
  208. void shmem_exit(struct shmem_fd *shmem_fd)
  209. {
  210. munmap(shmem_fd->linux_write, shmem_fd->shmem_len);
  211. close(shmem_fd->fd);
  212. }