1 /*M///////////////////////////////////////////////////////////////////////////////////////
2 //
3 // IMPORTANT: READ BEFORE DOWNLOADING, COPYING, INSTALLING OR USING.
4 //
5 // By downloading, copying, installing or using the software you agree to this license.
6 // If you do not agree to this license, do not download, install,
7 // copy or use the software.
8 //
9 //
10 // Intel License Agreement
11 // For Open Source Computer Vision Library
12 //
13 // Copyright (C) 2008, Nils Hasler, all rights reserved.
14 // Third party copyrights are property of their respective owners.
15 //
16 // Redistribution and use in source and binary forms, with or without modification,
17 // are permitted provided that the following conditions are met:
18 //
19 // * Redistribution's of source code must retain the above copyright notice,
20 // this list of conditions and the following disclaimer.
21 //
22 // * Redistribution's in binary form must reproduce the above copyright notice,
23 // this list of conditions and the following disclaimer in the documentation
24 // and/or other materials provided with the distribution.
25 //
26 // * The name of Intel Corporation may not be used to endorse or promote products
27 // derived from this software without specific prior written permission.
28 //
29 // This software is provided by the copyright holders and contributors "as is" and
30 // any express or implied warranties, including, but not limited to, the implied
31 // warranties of merchantability and fitness for a particular purpose are disclaimed.
32 // In no event shall the Intel Corporation or contributors be liable for any direct,
33 // indirect, incidental, special, exemplary, or consequential damages
34 // (including, but not limited to, procurement of substitute goods or services;
35 // loss of use, data, or profits; or business interruption) however caused
36 // and on any theory of liability, whether in contract, strict liability,
37 // or tort (including negligence or otherwise) arising in any way out of
38 // the use of this software, even if advised of the possibility of such damage.
39 //
40 //M*/
41
42 // Author: Bill McCord
43 //
44 // Intuitive Automata
45
46 //
47 // capture video from a socket connection
48 //
49
50 #define LOG_TAG "CVJNI"
51 #define LOGV(...) __android_log_print(ANDROID_LOG_SILENT, LOG_TAG, __VA_ARGS__)
52
53 #include "_highgui.h"
54 #include <android/log.h>
55 #include <errno.h>
56 #include <netdb.h>
57 #include <unistd.h>
58
59 #ifdef NDEBUG
60 #define CV_WARN(message)
61 #else
62 #define CV_WARN(message) fprintf(stderr, "warning: %s (%s:%d)\n", message, __FILE__, __LINE__)
63 #endif
64
65 #define IMAGE( i, x, y, n ) *(( unsigned char * )(( i )->imageData \
66 + ( x ) * sizeof( unsigned char ) * 3 \
67 + ( y ) * ( i )->widthStep ) + ( n ))
68
69 class CVCapture_Socket : public CvCapture
70 {
71 public:
CVCapture_Socket()72 CVCapture_Socket()
73 {
74 pAddrInfo = 0;
75 width = 0;
76 height = 0;
77 readBufSize = 0;
78 readBuf = 0;
79 frame = 0;
80 }
81
~CVCapture_Socket()82 virtual ~CVCapture_Socket()
83 {
84 close();
85 }
86
87 virtual bool open(const char* _address, const char* _port, int _width, int _height);
88 virtual void close();
89 virtual double getProperty(int);
90 virtual bool setProperty(int, double);
91 virtual bool grabFrame();
92 virtual IplImage* retrieveFrame();
93
94 protected:
95 struct addrinfo *pAddrInfo;
96 int width; // the width of the images received over the socket
97 int height; // the height of the images received over the socket
98 long readBufSize; // the length of the read buffer
99 char *readBuf; // the read buffer
100
101 IplImage* frame;
102 };
103
104 // The open method simply initializes some variables we will need later.
open(const char * _address,const char * _port,int _width,int _height)105 bool CVCapture_Socket::open(const char* _address, const char* _port, int _width, int _height)
106 {
107 // Free the addrinfo if it was allocated.
108 if (pAddrInfo)
109 {
110 freeaddrinfo(pAddrInfo);
111 pAddrInfo = 0;
112 }
113
114 // Check the easy stuff first.
115 width = _width;
116 height = _height;
117 if (width <= 0 || height <= 0)
118 {
119 LOGV("Invalid width or height!");
120 return false;
121 }
122
123 // Setup a new addrinfo to support a streaming socket at the given address and port.
124 struct addrinfo hints;
125 memset(&hints, 0, sizeof hints);
126 hints.ai_family = AF_UNSPEC; // use IPv4 or IPv6, whichever
127 hints.ai_socktype = SOCK_STREAM;
128 hints.ai_flags = AI_NUMERICHOST;
129
130 int error = getaddrinfo(_address, _port, &hints, &pAddrInfo);
131 if (error)
132 {
133 char buffer[100];
134 sprintf(buffer, "getaddrinfo error: %s", gai_strerror(error));
135 LOGV("%s", buffer);
136 freeaddrinfo(pAddrInfo);
137 pAddrInfo = 0;
138 return false;
139 }
140
141 readBufSize = width * height * sizeof(int);
142 readBuf = (char*)malloc(readBufSize);
143 if (!readBuf)
144 {
145 LOGV("out of memory error");
146 freeaddrinfo(pAddrInfo);
147 pAddrInfo = 0;
148 return false;
149 }
150
151 return true;
152 }
153
154 // Close cleans up all of our state and cached data.
close()155 void CVCapture_Socket::close()
156 {
157 LOGV("Setting simple vars to 0");
158 width = 0;
159 height = 0;
160 readBufSize = 0;
161
162 LOGV("Freeing Addr Info");
163 if (pAddrInfo)
164 {
165 freeaddrinfo(pAddrInfo);
166 pAddrInfo = 0;
167 }
168
169 LOGV("Freeing Buffer");
170 if (readBuf)
171 {
172 free(readBuf);
173 readBuf = 0;
174 }
175
176 LOGV("Releasing Image");
177 if (frame)
178 {
179 cvReleaseImage( &frame );
180 frame = 0;
181 }
182
183 LOGV("Done closing Capture Socket");
184 }
185
186 // Helper to load pixels from a byte stream received over a socket.
loadPixels(char * pixels,int width,int height)187 static IplImage* loadPixels(char* pixels, int width, int height) {
188
189 int x, y, pos, int_size = sizeof(int);
190 IplImage *img = cvCreateImage(cvSize(width, height), IPL_DEPTH_8U, 3);
191
192 for ( y = 0; y < height; y++ ) {
193 pos = y * width * int_size;
194 for ( x = 0; x < width; x++, pos += int_size ) {
195 // blue
196 IMAGE( img, x, y, 0 ) = pixels[pos + 3] & 0xFF;
197 // green
198 IMAGE( img, x, y, 1 ) = pixels[pos + 2] & 0xFF;
199 // red
200 IMAGE( img, x, y, 2 ) = pixels[pos + 1] & 0xFF;
201 }
202 }
203
204 return img;
205 }
206
207 // Grabs a frame (image) from a socket.
grabFrame()208 bool CVCapture_Socket::grabFrame()
209 {
210 // First ensure that our addrinfo and read buffer are allocated.
211 if (pAddrInfo == 0 || readBuf == 0)
212 {
213 LOGV("You haven't opened the socket capture yet!");
214 return false;
215 }
216
217 // Establish the socket.
218 int sockd = socket(pAddrInfo->ai_family, pAddrInfo->ai_socktype, pAddrInfo->ai_protocol);
219 if (sockd < 0 || errno != 0)
220 {
221 char buffer[100];
222 sprintf(buffer, "Failed to create socket, errno = %d", errno);
223 LOGV("%s", buffer);
224 ::close(sockd);
225 return false;
226 }
227
228 // Now connect to the socket.
229 if (connect(sockd, pAddrInfo->ai_addr, pAddrInfo->ai_addrlen) < 0 || errno != 0)
230 {
231 char buffer[100];
232 sprintf(buffer, "socket connection errorno = %d", errno);
233 LOGV("%s", buffer);
234 ::close(sockd);
235 return false;
236 }
237
238 // Release the image if it hasn't been already because we are going to overwrite it.
239 if (frame)
240 {
241 cvReleaseImage( &frame );
242 frame = 0;
243 }
244
245 // Read the socket until we have filled the data with the space allocated OR run
246 // out of data which we treat as an error.
247 long read_count, total_read = 0;
248 while (total_read < readBufSize)
249 {
250 read_count = read(sockd, &readBuf[total_read], readBufSize);
251 if (read_count <= 0 || errno != 0)
252 {
253 char buffer[100];
254 sprintf(buffer, "socket read errorno = %d", errno);
255 LOGV("%s", buffer);
256 break;
257 }
258 total_read += read_count;
259 }
260
261 // If we read all of the data we expected, we will load the frame from the pixels.
262 if (total_read == readBufSize)
263 {
264 frame = loadPixels(readBuf, width, height);
265 }
266 else
267 {
268 LOGV("full read of pixels failed");
269 }
270
271 // Close the socket and return the frame!
272 ::close(sockd);
273
274 return frame != 0;
275 }
276
retrieveFrame()277 IplImage* CVCapture_Socket::retrieveFrame()
278 {
279 return frame;
280 }
281
getProperty(int id)282 double CVCapture_Socket::getProperty(int id)
283 {
284 LOGV("unknown/unhandled property");
285 return 0;
286 }
287
setProperty(int id,double value)288 bool CVCapture_Socket::setProperty(int id, double value)
289 {
290 LOGV("unknown/unhandled property");
291 return false;
292 }
293
cvCreateCameraCapture_Socket(const char * address,const char * port,int width,int height)294 CvCapture* cvCreateCameraCapture_Socket( const char *address, const char *port, int width, int height )
295 {
296 CVCapture_Socket* capture = new CVCapture_Socket;
297 if ( capture-> open(address, port, width, height) )
298 return capture;
299
300 delete capture;
301 return 0;
302 }
303