Mercurial > mplayer.hg
view libvo/vo_yuv4mpeg.c @ 14167:d99bff888a20
pre5try2 release
author | diego |
---|---|
date | Fri, 17 Dec 2004 00:07:22 +0000 |
parents | 9c47f90ad94a |
children | 08399423fe30 |
line wrap: on
line source
/* * vo_yuv4mpeg.c, yuv4mpeg (mjpegtools) interface * * Thrown together by * Robert Kesterson <robertk@robertk.com> * Based on the pgm output plugin, the rgb2rgb postproc filter, divxdec, * and probably others. * * This is undoubtedly incomplete, inaccurate, or just plain wrong. :-) * * 2002/06/19 Klaus Stengel <Klaus.Stengel@asamnet.de> * - added support for interlaced output * Activate by using '-vo yuv4mpeg:interlaced' * or '-vo yuv4mpeg:interlaced_bf' if your source has * bottom fields first * - added some additional checks to catch problems * * 2002/04/17 Juergen Hammelmann <juergen.hammelmann@gmx.de> * - added support for output of subtitles * best, if you give option '-osdlevel 0' to mplayer for * no watching the seek+timer */ #include <stdio.h> #include <stdlib.h> #include <string.h> #include <unistd.h> #include <errno.h> #include <fcntl.h> #include "config.h" #include "video_out.h" #include "video_out_internal.h" #include "mp_msg.h" #include "help_mp.h" #include "sub.h" #include "fastmemcpy.h" #include "postproc/rgb2rgb.h" #include "libmpcodecs/vf_scale.h" static vo_info_t info = { "yuv4mpeg output for mjpegtools", "yuv4mpeg", "Robert Kesterson <robertk@robertk.com>", "" }; LIBVO_EXTERN (yuv4mpeg) static int image_width = 0; static int image_height = 0; static float image_fps = 0; static uint8_t *image = NULL; static uint8_t *image_y = NULL; static uint8_t *image_u = NULL; static uint8_t *image_v = NULL; static uint8_t *rgb_buffer = NULL; static uint8_t *rgb_line_buffer = NULL; static char *yuv_filename = NULL; static int using_format = 0; static FILE *yuv_out; static int write_bytes; #define Y4M_ILACE_NONE 'p' /* non-interlaced, progressive frame */ #define Y4M_ILACE_TOP_FIRST 't' /* interlaced, top-field first */ #define Y4M_ILACE_BOTTOM_FIRST 'b' /* interlaced, bottom-field first */ /* Set progressive mode as default */ static int config_interlace = Y4M_ILACE_NONE; #define Y4M_IS_INTERLACED (config_interlace != Y4M_ILACE_NONE) static uint32_t config(uint32_t width, uint32_t height, uint32_t d_width, uint32_t d_height, uint32_t fullscreen, char *title, uint32_t format) { if (image_width == width && image_height == height && image_fps == vo_fps && vo_config_count) return 0; if (vo_config_count) { mp_msg(MSGT_VO, MSGL_WARN, "Video formats differ (w:%i=>%i, h:%i=>%i, fps:%f=>%f), " "restarting output.\n", image_width, width, image_height, height, image_fps, vo_fps); uninit(); } image_height = height; image_width = width; image_fps = vo_fps; using_format = format; if (Y4M_IS_INTERLACED) { if (height % 4) { mp_msg(MSGT_VO,MSGL_FATAL, MSGTR_VO_YUV4MPEG_InterlacedHeightDivisibleBy4); return -1; } rgb_line_buffer = malloc(image_width * 3); if (!rgb_line_buffer) { mp_msg(MSGT_VO,MSGL_FATAL, MSGTR_VO_YUV4MPEG_InterlacedLineBufAllocFail); return -1; } if (using_format == IMGFMT_YV12) mp_msg(MSGT_VO,MSGL_WARN, MSGTR_VO_YUV4MPEG_InterlacedInputNotRGB); } if (width % 2) { mp_msg(MSGT_VO,MSGL_FATAL, MSGTR_VO_YUV4MPEG_WidthDivisibleBy2); return -1; } if(using_format != IMGFMT_YV12) { sws_rgb2rgb_init(get_sws_cpuflags()); rgb_buffer = malloc(image_width * image_height * 3); if (!rgb_buffer) { mp_msg(MSGT_VO,MSGL_FATAL, MSGTR_VO_YUV4MPEG_NoMemRGBFrameBuf); return -1; } } write_bytes = image_width * image_height * 3 / 2; image = malloc(write_bytes); yuv_out = fopen(yuv_filename ? yuv_filename : "stream.yuv", "wb"); if (!yuv_out || image == 0) { mp_msg(MSGT_VO,MSGL_FATAL, MSGTR_VO_YUV4MPEG_OutFileOpenError, yuv_filename ? yuv_filename : "stream.yuv"); return -1; } image_y = image; image_u = image_y + image_width * image_height; image_v = image_u + image_width * image_height / 4; // This isn't right. // But it should work as long as the file isn't interlaced // or otherwise unusual (the "Ip A0:0" part). /* At least the interlacing is ok now */ fprintf(yuv_out, "YUV4MPEG2 W%d H%d F%ld:%ld I%c A0:0\n", image_width, image_height, (long)(image_fps * 1000000.0), (long)1000000, config_interlace); fflush(yuv_out); return 0; } /* Only use when h divisable by 2! */ static void swap_fields(uint8_t *ptr, const int h, const int stride) { int i; for (i=0; i<h; i +=2) { memcpy(rgb_line_buffer , ptr + stride * i , stride); memcpy(ptr + stride * i , ptr + stride * (i+1), stride); memcpy(ptr + stride * (i+1), rgb_line_buffer , stride); } } static void draw_alpha(int x0, int y0, int w, int h, unsigned char *src, unsigned char *srca, int stride) { switch (using_format) { case IMGFMT_YV12: vo_draw_alpha_yv12(w, h, src, srca, stride, image + y0 * image_width + x0, image_width); break; case IMGFMT_BGR|24: case IMGFMT_RGB|24: if (config_interlace != Y4M_ILACE_BOTTOM_FIRST) vo_draw_alpha_rgb24(w, h, src, srca, stride, rgb_buffer + (y0 * image_width + x0) * 3, image_width * 3); else { swap_fields (rgb_buffer, image_height, image_width * 3); vo_draw_alpha_rgb24(w, h, src, srca, stride, rgb_buffer + (y0 * image_width + x0) * 3, image_width * 3); swap_fields (rgb_buffer, image_height, image_width * 3); } break; } } static void draw_osd(void) { vo_draw_text(image_width, image_height, draw_alpha); } static void deinterleave_fields(uint8_t *ptr, const int stride, const int img_height) { unsigned int i, j, k_start = 1, modv = img_height - 1; unsigned char *line_state = malloc(modv); for (i=0; i<modv; i++) line_state[i] = 0; line_state[0] = 1; while(k_start < modv) { i = j = k_start; memcpy(rgb_line_buffer, ptr + stride * i, stride); while (!line_state[j]) { line_state[j] = 1; i = j; j = j * 2 % modv; memcpy(ptr + stride * i, ptr + stride * j, stride); } memcpy(ptr + stride * i, rgb_line_buffer, stride); while(k_start < modv && line_state[k_start]) k_start++; } free(line_state); } static void vo_y4m_write(const void *ptr, const size_t num_bytes) { if (fwrite(ptr, 1, num_bytes, yuv_out) != num_bytes) mp_msg(MSGT_VO,MSGL_ERR, MSGTR_VO_YUV4MPEG_OutFileWriteError); } static int write_last_frame(void) { uint8_t *upper_y, *upper_u, *upper_v, *rgb_buffer_lower; int rgb_stride, uv_stride, field_height; unsigned int i, low_ofs; fprintf(yuv_out, "FRAME\n"); if (using_format != IMGFMT_YV12) { rgb_stride = image_width * 3; uv_stride = image_width / 2; if (Y4M_IS_INTERLACED) { field_height = image_height / 2; upper_y = image; upper_u = upper_y + image_width * field_height; upper_v = upper_u + image_width * field_height / 4; low_ofs = image_width * field_height * 3 / 2; rgb_buffer_lower = rgb_buffer + rgb_stride * field_height; /* Write Y plane */ for(i = 0; i < field_height; i++) { vo_y4m_write(upper_y + image_width * i, image_width); vo_y4m_write(upper_y + image_width * i + low_ofs, image_width); } /* Write U and V plane */ for(i = 0; i < field_height / 2; i++) { vo_y4m_write(upper_u + uv_stride * i, uv_stride); vo_y4m_write(upper_u + uv_stride * i + low_ofs, uv_stride); } for(i = 0; i < field_height / 2; i++) { vo_y4m_write(upper_v + uv_stride * i, uv_stride); vo_y4m_write(upper_v + uv_stride * i + low_ofs, uv_stride); } return VO_TRUE; /* Image written; We have to stop here */ } } /* Write progressive frame */ vo_y4m_write(image, write_bytes); return VO_TRUE; } static void flip_page (void) { uint8_t *upper_y, *upper_u, *upper_v, *rgb_buffer_lower; int rgb_stride, uv_stride, field_height; unsigned int i, low_ofs; fprintf(yuv_out, "FRAME\n"); if (using_format != IMGFMT_YV12) { rgb_stride = image_width * 3; uv_stride = image_width / 2; if (Y4M_IS_INTERLACED) { field_height = image_height / 2; upper_y = image; upper_u = upper_y + image_width * field_height; upper_v = upper_u + image_width * field_height / 4; low_ofs = image_width * field_height * 3 / 2; rgb_buffer_lower = rgb_buffer + rgb_stride * field_height; deinterleave_fields(rgb_buffer, rgb_stride, image_height); rgb24toyv12(rgb_buffer, upper_y, upper_u, upper_v, image_width, field_height, image_width, uv_stride, rgb_stride); rgb24toyv12(rgb_buffer_lower, upper_y + low_ofs, upper_u + low_ofs, upper_v + low_ofs, image_width, field_height, image_width, uv_stride, rgb_stride); /* Write Y plane */ for(i = 0; i < field_height; i++) { vo_y4m_write(upper_y + image_width * i, image_width); vo_y4m_write(upper_y + image_width * i + low_ofs, image_width); } /* Write U and V plane */ for(i = 0; i < field_height / 2; i++) { vo_y4m_write(upper_u + uv_stride * i, uv_stride); vo_y4m_write(upper_u + uv_stride * i + low_ofs, uv_stride); } for(i = 0; i < field_height / 2; i++) { vo_y4m_write(upper_v + uv_stride * i, uv_stride); vo_y4m_write(upper_v + uv_stride * i + low_ofs, uv_stride); } return; /* Image written; We have to stop here */ } rgb24toyv12(rgb_buffer, image_y, image_u, image_v, image_width, image_height, image_width, uv_stride, rgb_stride); } /* Write progressive frame */ vo_y4m_write(image, write_bytes); } static uint32_t draw_slice(uint8_t *srcimg[], int stride[], int w,int h,int x,int y) { int i; uint8_t *dst, *src = srcimg[0]; switch (using_format) { case IMGFMT_YV12: // copy Y: dst = image_y + image_width * y + x; for (i = 0; i < h; i++) { memcpy(dst, src, w); src += stride[0]; dst += image_width; } { // copy U + V: int imgstride = image_width >> 1; uint8_t *src1 = srcimg[1]; uint8_t *src2 = srcimg[2]; uint8_t *dstu = image_u + imgstride * (y >> 1) + (x >> 1); uint8_t *dstv = image_v + imgstride * (y >> 1) + (x >> 1); for (i = 0; i < h / 2; i++) { memcpy(dstu, src1 , w >> 1); memcpy(dstv, src2, w >> 1); src1 += stride[1]; src2 += stride[2]; dstu += imgstride; dstv += imgstride; } } break; case IMGFMT_BGR24: case IMGFMT_RGB24: dst = rgb_buffer + (image_width * y + x) * 3; for (i = 0; i < h; i++) { memcpy(dst, src, w * 3); src += stride[0]; dst += image_width * 3; } break; } return 0; } static uint32_t draw_frame(uint8_t * src[]) { switch(using_format) { case IMGFMT_YV12: // gets done in draw_slice break; case IMGFMT_BGR24: case IMGFMT_RGB24: memcpy(rgb_buffer, src[0], image_width * image_height * 3); break; } return 0; } static uint32_t query_format(uint32_t format) { if (Y4M_IS_INTERLACED) { /* When processing interlaced material we want to get the raw RGB * data and do the YV12 conversion ourselves to have the chrominance * information sampled correct. */ switch(format) { case IMGFMT_YV12: return VFCAP_CSP_SUPPORTED|VFCAP_OSD|VFCAP_ACCEPT_STRIDE; case IMGFMT_BGR|24: case IMGFMT_RGB|24: return VFCAP_CSP_SUPPORTED|VFCAP_CSP_SUPPORTED_BY_HW|VFCAP_OSD|VFCAP_ACCEPT_STRIDE; } } else { switch(format) { case IMGFMT_YV12: return VFCAP_CSP_SUPPORTED|VFCAP_CSP_SUPPORTED_BY_HW|VFCAP_OSD|VFCAP_ACCEPT_STRIDE; case IMGFMT_BGR|24: case IMGFMT_RGB|24: return VFCAP_CSP_SUPPORTED|VFCAP_OSD|VFCAP_ACCEPT_STRIDE; } } return 0; } // WARNING: config(...) also uses this static void uninit(void) { if(image) free(image); image = NULL; if(yuv_out) fclose(yuv_out); yuv_out = NULL; if(rgb_buffer) free(rgb_buffer); rgb_buffer = NULL; if(rgb_line_buffer) free(rgb_line_buffer); rgb_line_buffer = NULL; if (yuv_filename) free(yuv_filename); yuv_filename = NULL; image_width = 0; image_height = 0; image_fps = 0; } static void check_events(void) { } static uint32_t preinit(const char *arg) { if(arg) { int parse_err = 0; unsigned int parse_pos = 0; while (arg[parse_pos] && !parse_err) { if (strncmp (&arg[parse_pos], "interlaced", 10) == 0) { parse_pos += 10; config_interlace = Y4M_ILACE_TOP_FIRST; } else if (strncmp (&arg[parse_pos], "interlaced_bf", 13) == 0) { parse_pos += 13; config_interlace = Y4M_ILACE_BOTTOM_FIRST; } else if (strncmp (&arg[parse_pos], "file=", 5) == 0) { int file_len; parse_pos += 5; file_len = strcspn (&arg[parse_pos], ":"); if (file_len < 0) { parse_err = 1; break; } yuv_filename = malloc (file_len + 1); memcpy (yuv_filename, &arg[parse_pos], file_len); yuv_filename[file_len] = 0; parse_pos += file_len; } if (arg[parse_pos] == ':') parse_pos++; else if (arg[parse_pos]) parse_err = 1; } if (parse_err) { mp_msg(MSGT_VO,MSGL_FATAL, MSGTR_VO_YUV4MPEG_UnknownSubDev,arg); return -1; } } /* Inform user which output mode is used */ switch (config_interlace) { case Y4M_ILACE_TOP_FIRST: mp_msg(MSGT_VO,MSGL_STATUS, MSGTR_VO_YUV4MPEG_InterlacedTFFMode); break; case Y4M_ILACE_BOTTOM_FIRST: mp_msg(MSGT_VO,MSGL_STATUS, MSGTR_VO_YUV4MPEG_InterlacedBFFMode); break; default: mp_msg(MSGT_VO,MSGL_STATUS, MSGTR_VO_YUV4MPEG_ProgressiveMode); break; } return 0; } static uint32_t control(uint32_t request, void *data, ...) { switch (request) { case VOCTRL_QUERY_FORMAT: return query_format(*((uint32_t*)data)); case VOCTRL_DUPLICATE_FRAME: return write_last_frame(); } return VO_NOTIMPL; }