[svn-commits] rizzo: branch rizzo/astobj2 r77314 - /team/rizzo/astobj2/channels/chan_oss.c

SVN commits to the Digium repositories svn-commits at lists.digium.com
Thu Jul 26 11:34:56 CDT 2007


Author: rizzo
Date: Thu Jul 26 11:34:55 2007
New Revision: 77314

URL: http://svn.digium.com/view/asterisk?view=rev&rev=77314
Log:
more webcam/video bits.
Add code to open a v4l device and read from it,
plus some debugging code to show various info
(when a frame is ready, its size, and so on).
All what's missing now it the call to the encoding
routines, and possibly a secondary SDL window for
the local stream.

Modified:
    team/rizzo/astobj2/channels/chan_oss.c

Modified: team/rizzo/astobj2/channels/chan_oss.c
URL: http://svn.digium.com/view/asterisk/team/rizzo/astobj2/channels/chan_oss.c?view=diff&rev=77314&r1=77313&r2=77314
==============================================================================
--- team/rizzo/astobj2/channels/chan_oss.c (original)
+++ team/rizzo/astobj2/channels/chan_oss.c Thu Jul 26 11:34:55 2007
@@ -42,6 +42,7 @@
  * experimental support to decode a video session.
  */
 //#define DROP_PACKETS	5	// if set, simulate this percentage of lost video packets
+#define HAVE_V4L	1
 #define HAVE_SDL	1
 #define HAVE_FFMPEG	1
 
@@ -84,13 +85,6 @@
 #include "asterisk/musiconhold.h"
 #include "asterisk/app.h"
 
-#if HAVE_FFMPEG
-#include <ffmpeg/avcodec.h>
-#endif
-#if HAVE_SDL
-#include <SDL/SDL.h>
-#endif
-
 /* ringtones we use */
 #include "busy.h"
 #include "ringtone.h"
@@ -327,6 +321,9 @@
  * Examples for video encoding are at
 http://www.irisa.fr/texmex/people/dufouil/ffmpegdoxy/apiexample_8c-source.html
  */
+
+#include <ffmpeg/avcodec.h>
+#include <SDL/SDL.h>
 
 /* Structures for ffmpeg processing */
 /*
@@ -350,6 +347,19 @@
 	int received;
 
 	struct ast_frame *echo;
+
+	/* webcam support.
+	 * videodevice and geometry are read from the config file.
+	 * At the right time we try to open it and allocate a buffer.
+	 * If we are successful, webcam_bufsize > 0 and we can read.
+	 */
+	const char *videodevice;	/* pointer in the parent struct */
+	int w;			/* geometry */
+	int h;
+	int webcam_fd;		/* file descriptor */
+	int webcam_bufsize;	/* buffer size, also total bytes per frame */
+	int webcam_ofs;		/* offset for next read */
+	char *webcam_imgbuf;	/* malloced area of size webcam_bufsize */
 };
 
 struct _cm {	/* map ffmpeg codec types to asterisk formats */
@@ -361,6 +371,124 @@
 	{ AST_FORMAT_H263_PLUS,	CODEC_ID_H263 },
 	{ 0,			0 },
 };
+
+/*! \brief support to open a webcam using video4linux1 and read()
+ *
+ */
+#if !defined(HAVE_V4L) || HAVE_V4L != 1
+/* stubs */
+static int webcam_open(struct video_desc *env)
+{
+	return -1;
+}
+static int webcam_read(struct video_desc *env)
+{
+	return 0;
+}
+
+#else
+
+/* the real thing */
+#include <linux/videodev.h>
+
+static int webcam_open(struct video_desc *env)
+{
+	int fd, i;
+	const char *device = env->videodevice;
+	struct video_window vw = { 0 };	/* camera attributes */
+	struct video_picture vp;
+
+	int bufsize;
+	char *imgbuf = NULL;
+
+	if (env->webcam_imgbuf)
+		return env->webcam_fd;
+
+	fd = open(device, O_RDONLY | O_NONBLOCK);
+	if (fd < 0) {
+		ast_log(LOG_WARNING, "error opening camera %s\n", device);
+		return fd;
+	}
+
+	i = fcntl(fd, F_GETFL);
+	if (-1 == fcntl(fd, F_GETFL, i | O_NONBLOCK)) {
+		ast_log(LOG_WARNING, "error F_SETFL for %s\n", device);
+		goto error;
+	}
+	/* set format */
+	vw.width = env->w;
+	vw.height = env->h;
+	bufsize = (vw.width * vw.height * 3)/2;	/* yuv411 */
+	if (ioctl(fd, VIDIOCSWIN, &vw) == -1) {
+		ast_log(LOG_WARNING, "error setting format for %s\n", device);
+		goto error;
+	}
+	if (ioctl(fd, VIDIOCGPICT, &vp) == -1) {
+                ast_log(LOG_WARNING, "error reading picture info\n");
+                return -1;
+        }
+	ast_log(LOG_WARNING, "contrast %d bright %d colour %d hue %d whiteness %d palette %d\n",
+                vp.contrast, vp.brightness,
+                vp.colour, vp.hue,
+                vp.whiteness, vp.palette);
+	vp.palette = VIDEO_PALETTE_YUV420P;
+	if(ioctl(fd,VIDIOCSPICT,&vp) == -1) {
+		ast_log(LOG_WARNING, "error setting picture info\n");
+		goto error;
+        }
+	imgbuf = malloc(bufsize);
+	if (!imgbuf)
+		goto error;
+
+	env->webcam_fd = fd;
+	env->webcam_bufsize = bufsize;
+	env->webcam_ofs = 0;
+	env->webcam_imgbuf = imgbuf;
+	ast_log(LOG_WARNING, "videodev %s opened, size %dx%d %d\n", device, env->w, env->h, bufsize);
+	return fd;
+
+error:
+	if (fd >= 0)
+		close(fd);
+	if (imgbuf)
+		free(imgbuf);
+	env->webcam_fd = -1;
+	return -1;
+}
+
+/*! \brief complete a buffer from the webcam */
+static int webcam_read(struct video_desc *env)
+{
+	if (env->webcam_bufsize == 0)
+		return 0;
+
+	for (;;) {
+		int r, l = env->webcam_bufsize - env->webcam_ofs;
+		r = read(env->webcam_fd, env->webcam_imgbuf + env->webcam_ofs, l);
+		// ast_log(LOG_WARNING, "read %d of %d bytes from webcam\n", r, l);
+		if (r < 0) {	/* read error */
+			return 0;
+		}
+		if (r == 0)	/* no data */
+			return 0;
+		env->webcam_ofs += r;
+		if (r == l) {
+			env->webcam_ofs = 0; /* prepare for next frame */
+			return env->webcam_bufsize;
+		}
+	}
+}
+#endif /* HAVE_V4L */
+
+static void show_frame(struct video_desc *env);
+
+static void webcam_encode(struct video_desc *env)
+{
+	struct timeval now = ast_tvnow();
+	ast_log(LOG_WARNING, "video frame ready at %d.%06d\n",
+		(int)now.tv_sec % 1000, (int)now.tv_usec);
+	show_frame(env);
+}
 
 /* Helper function to process incoming video.
  * For each incoming video call invoke ffmpeg_init() to intialize
@@ -372,18 +500,11 @@
  * to display the frame.
  *
  */
-/* Extract the bitstream from the RTP payload */
-static uint8_t *pre_process_data(uint8_t *, int *);
-/* Decode video frame once completed */
-static int decode_video(struct video_desc *);
-/* Dispaly decoded frame */
-static void show_frame(struct video_desc *);
 
 static struct video_desc *get_video_desc(struct ast_channel *c);
 
 /* Macros used as a wrapper around the actual video format we want to use */
 #define AST_FORMAT_CUSTOM (AST_FORMAT_H263_PLUS)
-static int write_video(struct ast_channel *chan, struct ast_frame *f);
 
 /*! \brief map an asterisk format into an ffmpeg one */
 static enum CodecID map_video_format(uint32_t ast_format)
@@ -498,8 +619,21 @@
 	if(env->bmp)
 		SDL_FreeYUVOverlay(env->bmp);
 	SDL_Quit();
-	bzero(env, sizeof(struct video_desc));
-	env->initialized = 0;
+
+	if (env->webcam_imgbuf) {
+		free(env->webcam_imgbuf);
+		if (env->webcam_fd >= 0)
+		close(env->webcam_fd);
+	}
+	{	/* clear the struct but restore some fields */
+		const char *vd = env->videodevice;
+		int w = env->w, h = env->h;
+		bzero(env, sizeof(struct video_desc));
+		/* restore fields... */
+		env->videodevice = vd;
+		env->w = w;
+		env->h = h;
+	}
 }
 
 #define MAKE_MASK(bits)                ( (1<<(bits)) -1 )
@@ -585,6 +719,8 @@
 	uint8_t *data;
 	int datalen;
 
+	if (!len)
+		return 0;
 	while(len) {
 		ret = av_parser_parse(env->parser, env->context, &data, &datalen, aux, len, 0, 0);
 		if(datalen) {
@@ -613,6 +749,8 @@
 	AVPicture pict;
 	SDL_Rect rect;
 
+	if (!env->initialized)
+		return;
 	if(env->screen == NULL) {
 		env->screen = SDL_SetVideoMode(env->context->width, env->context->height, 0, 0);
 		if(!env->screen) {
@@ -638,7 +776,12 @@
 		(AVPicture *)env->frame, env->context->pix_fmt,
 		env->context->width, env->context->height);
 	SDL_UnlockYUVOverlay(env->bmp);
-
+#if 0	/* more testing, overlay the received image with the local picture */
+	ast_log(LOG_WARNING, "show_frame: linesize %d %d %d\n", pict.linesize[0], pict.linesize[1], pict.linesize[2]);
+	if (env->webcam_imgbuf) {
+		bcopy(env->webcam_imgbuf, pict.data[0], env->context->width*env->context->height);
+	}
+#endif
 	rect.x = 0; rect.y = 0;
 	rect.w = env->context->width;
 	rect.h = env->context->height;
@@ -661,16 +804,19 @@
 	struct video_desc *env = get_video_desc(chan);
 	struct timeval now = ast_tvnow();
 	int i;
-	struct ast_frame *f1, **fp;
 
 	if(!env->initialized)
 		ffmpeg_init(env, f->subclass);
 	if(!env->initialized)
 		return -1;	/* error */
-	f1 = ast_frdup(f);
-	for (fp = &env->echo; (*fp) != NULL ; fp = &AST_LIST_NEXT((*fp), frame_list) )
-		;
-	*fp = f1;
+
+	if (0) {	/* echo frames back to the sender */
+		struct ast_frame *f1, **fp;
+		f1 = ast_frdup(f);
+		for (fp = &env->echo; (*fp) != NULL ; fp = &AST_LIST_NEXT((*fp), frame_list) )
+			;
+		*fp = f1;
+	}
 
 	i = ast_tvdiff_ms(now, env->ts);
 	if (i > 1000) {
@@ -838,6 +984,7 @@
 	int readpos;				/*!< read position above */
 	struct ast_frame read_f;	/*!< returned by oss_read */
 
+	char videodevice[64];
 #if HAVE_FFMPEG
 	struct video_desc env;
 #endif
@@ -1237,6 +1384,8 @@
 	res = PCM_ENABLE_INPUT | PCM_ENABLE_OUTPUT;
 	res = ioctl(fd, SNDCTL_DSP_SETTRIGGER, &res);
 	/* it may fail if we are in half duplex, never mind */
+
+	fd = webcam_open(&o->env);
 	return 0;
 }
 
@@ -1402,6 +1551,9 @@
 	struct timeval tb;
 
 	struct video_desc *env = get_video_desc(c);
+
+	if (webcam_read(env) && 0)
+		webcam_encode(env);	/* get frames and put them in the queue */
 	if (env->echo) {
 		struct ast_frame *f1;
 		int i = 0;
@@ -2039,9 +2191,13 @@
 		M_F("mixer", store_mixer(o, v->value))
 		M_F("callerid", store_callerid(o, v->value))
 		M_F("boost", store_boost(o, v->value))
+		M_STR("videodevice", o->videodevice)
+		M_UINT("videowidth", o->env.w)
+		M_UINT("videoheight", o->env.h)
 
 		M_END(/* */);
 	}
+	o->env.videodevice = o->videodevice;
 	if (ast_strlen_zero(o->device))
 		ast_copy_string(o->device, DEV_DSP, sizeof(o->device));
 	if (o->mixer_cmd) {




More information about the svn-commits mailing list