You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
			
				
					195 lines
				
				4.4 KiB
			
		
		
			
		
	
	
					195 lines
				
				4.4 KiB
			| 
											6 years ago
										 | #include <stdio.h>
 | ||
|  | #include <stdlib.h>
 | ||
|  | #include <stdbool.h>
 | ||
|  | #include <string.h>
 | ||
|  | #include <unistd.h>
 | ||
|  | #include <assert.h>
 | ||
|  | #include <errno.h>
 | ||
|  | 
 | ||
|  | #include <sys/mman.h>
 | ||
|  | #include <sys/socket.h>
 | ||
|  | #include <sys/un.h>
 | ||
|  | 
 | ||
|  | #include "ipc.h"
 | ||
|  | 
 | ||
|  | #include "visionipc.h"
 | ||
|  | 
 | ||
|  | typedef struct VisionPacketWire {
 | ||
|  |   int type;
 | ||
|  |   VisionPacketData d;
 | ||
|  | } VisionPacketWire;
 | ||
|  | 
 | ||
|  | int vipc_connect() {
 | ||
|  |   return ipc_connect(VIPC_SOCKET_PATH);
 | ||
|  | }
 | ||
|  | 
 | ||
|  | 
 | ||
|  | int vipc_recv(int fd, VisionPacket *out_p) {
 | ||
|  |   VisionPacketWire p = {0};
 | ||
|  |   VisionPacket p2 = {0};
 | ||
|  |   int ret = ipc_sendrecv_with_fds(false, fd, &p, sizeof(p), (int*)p2.fds, VIPC_MAX_FDS, &p2.num_fds);
 | ||
|  |   if (ret < 0) {
 | ||
|  |     printf("vipc_recv err: %s\n", strerror(errno));
 | ||
|  |   } else {
 | ||
|  |     p2.type = p.type;
 | ||
|  |     p2.d = p.d;
 | ||
|  |     *out_p = p2;
 | ||
|  |   }
 | ||
|  |   //printf("%d = vipc_recv(%d, %d): %d %d %d %u\n", ret, fd, p2.num_fds, out_p->d.stream_bufs.type, out_p->d.stream_bufs.width, out_p->d.stream_bufs.height, out_p->d.stream_bufs.buf_len);
 | ||
|  |   return ret;
 | ||
|  | }
 | ||
|  | 
 | ||
|  | int vipc_send(int fd, const VisionPacket *p2) {
 | ||
|  |   assert(p2->num_fds <= VIPC_MAX_FDS);
 | ||
|  | 
 | ||
|  |   VisionPacketWire p = {
 | ||
|  |     .type = p2->type,
 | ||
|  |     .d = p2->d,
 | ||
|  |   };
 | ||
|  |   int ret = ipc_sendrecv_with_fds(true, fd, (void*)&p, sizeof(p), (int*)p2->fds, p2->num_fds, NULL);
 | ||
|  |   //printf("%d = vipc_send(%d, %d): %d %d %d %u\n", ret, fd, p2->num_fds, p2->d.stream_bufs.type, p2->d.stream_bufs.width, p2->d.stream_bufs.height, p2->d.stream_bufs.buf_len);
 | ||
|  |   return ret;
 | ||
|  | }
 | ||
|  | 
 | ||
|  | void vipc_bufs_load(VIPCBuf *bufs, const VisionStreamBufs *stream_bufs,
 | ||
|  |                      int num_fds, const int* fds) {
 | ||
|  |   for (int i=0; i<num_fds; i++) {
 | ||
|  |     if (bufs[i].addr) {
 | ||
|  |       munmap(bufs[i].addr, bufs[i].len);
 | ||
|  |       bufs[i].addr = NULL;
 | ||
|  |       close(bufs[i].fd);
 | ||
|  |     }
 | ||
|  |     bufs[i].fd = fds[i];
 | ||
|  |     bufs[i].len = stream_bufs->buf_len;
 | ||
|  |     bufs[i].addr = mmap(NULL, bufs[i].len,
 | ||
|  |                         PROT_READ | PROT_WRITE,
 | ||
|  |                         MAP_SHARED, bufs[i].fd, 0);
 | ||
|  |     // printf("b %d %zu -> %p\n", bufs[i].fd, bufs[i].len, bufs[i].addr);
 | ||
|  |     assert(bufs[i].addr != MAP_FAILED);
 | ||
|  |   }
 | ||
|  | }
 | ||
|  | 
 | ||
|  | 
 | ||
|  | int visionstream_init(VisionStream *s, VisionStreamType type, bool tbuffer, VisionStreamBufs *out_bufs_info) {
 | ||
|  |   int err;
 | ||
|  | 
 | ||
|  |   memset(s, 0, sizeof(*s));
 | ||
|  | 
 | ||
|  |   s->last_idx = -1;
 | ||
|  | 
 | ||
|  |   s->ipc_fd = vipc_connect();
 | ||
|  |   if (s->ipc_fd < 0) return -1;
 | ||
|  | 
 | ||
|  |   VisionPacket p = {
 | ||
|  |     .type = VIPC_STREAM_SUBSCRIBE,
 | ||
|  |     .d = { .stream_sub = {
 | ||
|  |       .type = type,
 | ||
|  |       .tbuffer = tbuffer,
 | ||
|  |     }, },
 | ||
|  |   };
 | ||
|  |   err = vipc_send(s->ipc_fd, &p);
 | ||
|  |   if (err < 0) {
 | ||
|  |     close(s->ipc_fd);
 | ||
|  |     return -1;
 | ||
|  |   }
 | ||
|  | 
 | ||
|  |   VisionPacket rp;
 | ||
|  |   err = vipc_recv(s->ipc_fd, &rp);
 | ||
|  |   if (err <= 0) {
 | ||
|  |     close(s->ipc_fd);
 | ||
|  |     return -1;
 | ||
|  |   }
 | ||
|  |   assert(rp.type = VIPC_STREAM_BUFS);
 | ||
|  |   assert(rp.d.stream_bufs.type == type);
 | ||
|  | 
 | ||
|  |   s->bufs_info = rp.d.stream_bufs;
 | ||
|  | 
 | ||
|  |   s->num_bufs = rp.num_fds;
 | ||
|  |   s->bufs = calloc(s->num_bufs, sizeof(VIPCBuf));
 | ||
|  |   assert(s->bufs);
 | ||
|  | 
 | ||
|  |   vipc_bufs_load(s->bufs, &rp.d.stream_bufs, s->num_bufs, rp.fds);
 | ||
|  | 
 | ||
|  |   if (out_bufs_info) {
 | ||
|  |     *out_bufs_info = s->bufs_info;
 | ||
|  |   }
 | ||
|  | 
 | ||
|  |   return 0;
 | ||
|  | }
 | ||
|  | 
 | ||
|  | void visionstream_release(VisionStream *s) {
 | ||
|  |   int err;
 | ||
|  |   if (s->last_idx >= 0) {
 | ||
|  |     VisionPacket rep = {
 | ||
|  |       .type = VIPC_STREAM_RELEASE,
 | ||
|  |       .d = { .stream_rel = {
 | ||
|  |         .type = s->last_type,
 | ||
|  |         .idx = s->last_idx,
 | ||
|  |       }}
 | ||
|  |     };
 | ||
|  |     err = vipc_send(s->ipc_fd, &rep);
 | ||
|  |     s->last_idx = -1;
 | ||
|  |   }
 | ||
|  | }
 | ||
|  | 
 | ||
|  | VIPCBuf* visionstream_get(VisionStream *s, VIPCBufExtra *out_extra) {
 | ||
|  |   int err;
 | ||
|  | 
 | ||
|  |   VisionPacket rp;
 | ||
|  |   err = vipc_recv(s->ipc_fd, &rp);
 | ||
|  |   if (err <= 0) {
 | ||
|  |     return NULL;
 | ||
|  |   }
 | ||
|  |   assert(rp.type == VIPC_STREAM_ACQUIRE);
 | ||
|  | 
 | ||
|  |   if (s->last_idx >= 0) {
 | ||
|  |     VisionPacket rep = {
 | ||
|  |       .type = VIPC_STREAM_RELEASE,
 | ||
|  |       .d = { .stream_rel = {
 | ||
|  |         .type = s->last_type,
 | ||
|  |         .idx = s->last_idx,
 | ||
|  |       }}
 | ||
|  |     };
 | ||
|  |     err = vipc_send(s->ipc_fd, &rep);
 | ||
|  |     if (err <= 0) {
 | ||
|  |       return NULL;
 | ||
|  |     }
 | ||
|  |   }
 | ||
|  | 
 | ||
|  |   s->last_type = rp.d.stream_acq.type;
 | ||
|  |   s->last_idx = rp.d.stream_acq.idx;
 | ||
|  |   assert(s->last_idx < s->num_bufs);
 | ||
|  | 
 | ||
|  |   if (out_extra) {
 | ||
|  |     *out_extra = rp.d.stream_acq.extra;
 | ||
|  |   }
 | ||
|  | 
 | ||
|  |   return &s->bufs[s->last_idx];
 | ||
|  | }
 | ||
|  | 
 | ||
|  | void visionstream_destroy(VisionStream *s) {
 | ||
|  |   int err;
 | ||
|  | 
 | ||
|  |   if (s->last_idx >= 0) {
 | ||
|  |     VisionPacket rep = {
 | ||
|  |       .type = VIPC_STREAM_RELEASE,
 | ||
|  |       .d = { .stream_rel = {
 | ||
|  |         .type = s->last_type,
 | ||
|  |         .idx = s->last_idx,
 | ||
|  |       }}
 | ||
|  |     };
 | ||
|  |     err = vipc_send(s->ipc_fd, &rep);
 | ||
|  |     s->last_idx = -1;
 | ||
|  |   }
 | ||
|  | 
 | ||
|  |   for (int i=0; i<s->num_bufs; i++) {
 | ||
|  |     if (s->bufs[i].addr) {
 | ||
|  |       munmap(s->bufs[i].addr, s->bufs[i].len);
 | ||
|  |       s->bufs[i].addr = NULL;
 | ||
|  |       close(s->bufs[i].fd);
 | ||
|  |     }
 | ||
|  |   }
 | ||
|  |   if (s->bufs) free(s->bufs);
 | ||
|  |   close(s->ipc_fd);
 | ||
|  | }
 |