author | sandman <sandman> | 2002-08-04 20:23:19 (UTC) |
---|---|---|
committer | sandman <sandman> | 2002-08-04 20:23:19 (UTC) |
commit | 57bd412cf973805fbe69ecfa8f168ad2e28311a9 (patch) (side-by-side diff) | |
tree | 78d7bab924023bdf33a437447bb31fff52b51c32 | |
parent | 7bf26dd95a7bd434edc8dd5e001d8ac490f67dc3 (diff) | |
download | opie-57bd412cf973805fbe69ecfa8f168ad2e28311a9.zip opie-57bd412cf973805fbe69ecfa8f168ad2e28311a9.tar.gz opie-57bd412cf973805fbe69ecfa8f168ad2e28311a9.tar.bz2 |
- Removed the mlib and mmx yuv2rgb converters
- Added an optimized (non-scaling !) arm4l yuv2rgb (taken from bbplay)
-rw-r--r-- | noncore/multimedia/opieplayer2/nullvideo.c | 2 | ||||
-rw-r--r-- | noncore/multimedia/opieplayer2/opieplayer2.pro | 2 | ||||
-rw-r--r-- | noncore/multimedia/opieplayer2/yuv2rgb.c | 8 | ||||
-rw-r--r-- | noncore/multimedia/opieplayer2/yuv2rgb_arm.c | 174 | ||||
-rw-r--r-- | noncore/multimedia/opieplayer2/yuv2rgb_arm4l.S | 192 | ||||
-rw-r--r-- | noncore/multimedia/opieplayer2/yuv2rgb_mlib.c | 313 | ||||
-rw-r--r-- | noncore/multimedia/opieplayer2/yuv2rgb_mmx.c | 1047 |
7 files changed, 375 insertions, 1363 deletions
diff --git a/noncore/multimedia/opieplayer2/nullvideo.c b/noncore/multimedia/opieplayer2/nullvideo.c index 79337c2..bd52869 100644 --- a/noncore/multimedia/opieplayer2/nullvideo.c +++ b/noncore/multimedia/opieplayer2/nullvideo.c @@ -1,674 +1,672 @@ /*#include <xine.h>*/ #include <stdlib.h> #include <stdio.h> #include <math.h> #include <xine/video_out.h> #include <xine/xine_internal.h> #include <xine/xineutils.h> #include <xine/configfile.h> #include <pthread.h> #include "alphablend.h" #include "yuv2rgb.h" #define printf(x,...) /* the caller for our event draw handler */ typedef void (*display_xine_frame_t) (void *user_data, uint8_t* frame, int width, int height,int bytes ); typedef struct null_driver_s null_driver_t; struct null_driver_s { vo_driver_t vo_driver; uint32_t m_capabilities; int m_show_video; int m_video_fullscreen; int m_is_scaling; int depth, bpp, bytes_per_pixel; int yuv2rgb_mode; int yuv2rgb_swap; int zuv2rgb_gamma; uint8_t *yuv2rgb_cmap; yuv2rgb_factory_t *yuv2rgb_factory; vo_overlay_t *overlay; int user_ratio; double output_scale_factor; int last_frame_output_width; int last_frame_output_height; int gui_width; int gui_height; int gui_changed; double display_ratio; void* caller; display_xine_frame_t frameDis; }; typedef struct opie_frame_s opie_frame_t; struct opie_frame_s { vo_frame_t frame; char* name; int version; int width; int height; int ratio_code; int format; int flags; int user_ratio; double ratio_factor; int ideal_width; int ideal_height; int output_width, output_height; int gui_width, gui_height; uint8_t *chunk[3]; yuv2rgb_t *yuv2rgb; uint8_t *rgb_dst; int yuv_stride; int stripe_height, stripe_inc; int bytes_per_line; uint8_t *data; // int show_video; null_driver_t *output; }; static uint32_t null_get_capabilities(vo_driver_t *self ){ null_driver_t* this = (null_driver_t*)self; printf("capabilities\n"); return this->m_capabilities; } static void null_frame_copy (vo_frame_t *vo_img, uint8_t **src) { opie_frame_t *frame = (opie_frame_t *) vo_img ; printf("frame copy\n"); if(!frame->output->m_show_video ){ printf("no video\n"); return; } // no video if (frame->format == IMGFMT_YV12) { frame->yuv2rgb->yuv2rgb_fun (frame->yuv2rgb, frame->rgb_dst, src[0], src[1], src[2]); } else { frame->yuv2rgb->yuy22rgb_fun (frame->yuv2rgb, frame->rgb_dst, src[0]); } frame->rgb_dst += frame->stripe_inc; printf("returning\n"); } static void null_frame_field (vo_frame_t *vo_img, int which_field) { opie_frame_t *frame = (opie_frame_t *) vo_img ; printf("field\n\n"); switch (which_field) { case VO_TOP_FIELD: frame->rgb_dst = (uint8_t *)frame->data; frame->stripe_inc = 2*frame->stripe_height * frame->bytes_per_line; break; case VO_BOTTOM_FIELD: frame->rgb_dst = (uint8_t *)frame->data + frame->bytes_per_line ; frame->stripe_inc = 2*frame->stripe_height * frame->bytes_per_line; break; case VO_BOTH_FIELDS: frame->rgb_dst = (uint8_t *)frame->data; break; } } /* take care of the frame*/ static void null_frame_dispose( vo_frame_t* vo_img){ opie_frame_t* frame = (opie_frame_t*)vo_img; printf("frame_dispose\n"); if( frame->data ) free( frame->data ); free (frame); } /* end take care of frames*/ static vo_frame_t* null_alloc_frame( vo_driver_t* self ){ null_driver_t* this = (null_driver_t*)self; opie_frame_t* frame; frame = (opie_frame_t*)malloc ( sizeof(opie_frame_t) ); memset( frame, 0, sizeof( opie_frame_t) ); pthread_mutex_init (&frame->frame.mutex, NULL); printf("alloc_frame\n"); frame->name = "opie\0"; frame->version = 1; frame->output = this; // frame->show_video = this->m_show_video; /* initialize the frame*/ frame->frame.driver = self; /*frame.frame.free = null_frame_free;*/ frame->frame.copy = null_frame_copy; frame->frame.field = null_frame_field; frame->frame.dispose = null_frame_dispose; frame->yuv2rgb = 0; /* * colorspace converter for this frame */ frame->yuv2rgb = this->yuv2rgb_factory->create_converter (this->yuv2rgb_factory); return (vo_frame_t*) frame; } // size specific static void null_compute_ideal_size (null_driver_t *this, opie_frame_t *frame) { if (!this->m_is_scaling /*|| !this->m_show_video*/) { printf("Not scaling\n"); frame->ideal_width = frame->width; frame->ideal_height = frame->height; frame->ratio_factor = 1.0; } else { double image_ratio, desired_ratio, corr_factor; image_ratio = (double) frame->width / (double) frame->height; switch (frame->user_ratio) { case ASPECT_AUTO: switch (frame->ratio_code) { case XINE_ASPECT_RATIO_ANAMORPHIC: /* anamorphic */ case XINE_ASPECT_RATIO_PAN_SCAN: desired_ratio = 16.0 /9.0; break; case XINE_ASPECT_RATIO_211_1: /* 2.11:1 */ desired_ratio = 2.11/1.0; break; case XINE_ASPECT_RATIO_SQUARE: /* square pels */ case XINE_ASPECT_RATIO_DONT_TOUCH: /* probably non-mpeg stream => don't touch aspect ratio */ desired_ratio = image_ratio; break; case 0: /* forbidden -> 4:3 */ printf ("video_out_fb: invalid ratio, using 4:3\n"); default: printf ("video_out_fb: unknown aspect ratio (%d) in stream => using 4:3\n", frame->ratio_code); case XINE_ASPECT_RATIO_4_3: /* 4:3 */ desired_ratio = 4.0 / 3.0; break; } break; case ASPECT_ANAMORPHIC: desired_ratio = 16.0 / 9.0; break; case ASPECT_DVB: desired_ratio = 2.0 / 1.0; break; case ASPECT_SQUARE: desired_ratio = image_ratio; break; case ASPECT_FULL: default: desired_ratio = 4.0 / 3.0; } frame->ratio_factor = this->display_ratio * desired_ratio; corr_factor = frame->ratio_factor / image_ratio ; if (fabs(corr_factor - 1.0) < 0.005) { frame->ideal_width = frame->width; frame->ideal_height = frame->height; } else { if (corr_factor >= 1.0) { frame->ideal_width = frame->width * corr_factor + 0.5; frame->ideal_height = frame->height; } else { frame->ideal_width = frame->width; frame->ideal_height = frame->height / corr_factor + 0.5; } } } printf("return from helper\n"); } static void null_compute_rgb_size (null_driver_t *this, opie_frame_t *frame) { double x_factor, y_factor; /* * make the frame fit into the given destination area */ x_factor = (double) this->gui_width / (double) frame->ideal_width; y_factor = (double) this->gui_height / (double) frame->ideal_height; if ( x_factor < y_factor ) { frame->output_width = (double) frame->ideal_width * x_factor ; frame->output_height = (double) frame->ideal_height * x_factor ; } else { frame->output_width = (double) frame->ideal_width * y_factor ; frame->output_height = (double) frame->ideal_height * y_factor ; } #define LOG 1 #ifdef LOG printf("video_out_fb: frame source %d x %d => screen output %d x %d%s\n", frame->width, frame->height, frame->output_width, frame->output_height, ( frame->width != frame->output_width || frame->height != frame->output_height ? ", software scaling" : "" ) ); #endif } // size specific static void null_update_frame_format( vo_driver_t* self, vo_frame_t* img, uint32_t width, uint32_t height, int ratio_code, int format, int flags ){ null_driver_t* this = (null_driver_t*) self; opie_frame_t* frame = (opie_frame_t*)img; /* not needed now */ printf("update_frame_format\n"); printf("al crash aye?\n"); flags &= VO_BOTH_FIELDS; /* find out if we need to adapt this frame */ if ((width != frame->width) || (height != frame->height) || (ratio_code != frame->ratio_code) || (flags != frame->flags) || (format != frame->format) || (this->user_ratio != frame->user_ratio) || (this->gui_width != frame-> gui_width ) || (this-> gui_height != frame-> gui_height)) { frame->width = width; frame->height = height; frame->ratio_code = ratio_code; frame->flags = flags; frame->format = format; frame->user_ratio = this->user_ratio; this->gui_changed = 0; // frame->show_video = this->m_show_video; frame->gui_width = this->gui_width; frame->gui_height = this->gui_height; null_compute_ideal_size (this, frame); null_compute_rgb_size (this, frame); /* * (re-) allocate */ if( frame->data ) { if(frame->chunk[0] ){ free( frame->chunk[0] ); frame->chunk[0] = NULL; } if(frame->chunk[1] ){ free ( frame->chunk[1] ); frame->chunk[1] = NULL; } if(frame->chunk[2] ){ free ( frame->chunk[2] ); frame->chunk[2] = NULL; } free ( frame->data ); } printf("after freeing\n"); frame->data = xine_xmalloc (frame->output_width * frame->output_height * this->bytes_per_pixel ); if( format == IMGFMT_YV12 ) { frame->frame.pitches[0] = 8*((width + 7) / 8); frame->frame.pitches[1] = 8*((width + 15) / 16); frame->frame.pitches[2] = 8*((width + 15) / 16); frame->frame.base[0] = xine_xmalloc_aligned (16, frame->frame.pitches[0] * height,(void **)&frame->chunk[0]); frame->frame.base[1] = xine_xmalloc_aligned (16, frame->frame.pitches[1] * ((height+ 1)/2), (void **)&frame->chunk[1]); frame->frame.base[2] = xine_xmalloc_aligned (16, frame->frame.pitches[2] * ((height+ 1)/2), (void **)&frame->chunk[2]); }else{ frame->frame.pitches[0] = 8*((width + 3) / 4); - frame->frame.pitches[1] = 8*((width + 3) / 4); - frame->frame.pitches[2] = 8*((width + 3) / 4); frame->frame.base[0] = xine_xmalloc_aligned (16, frame->frame.pitches[0] * height, (void **)&frame->chunk[0]); frame->chunk[1] = NULL; frame->chunk[2] = NULL; } frame->format = format; frame->width = width; frame->height = height; frame->stripe_height = 16 * frame->output_height / frame->height; frame->bytes_per_line = frame->output_width * this->bytes_per_pixel; /* * set up colorspace converter */ if(1 /*this->m_show_video*/ ){ printf("showing video\n"); switch (flags) { case VO_TOP_FIELD: case VO_BOTTOM_FIELD: frame->yuv2rgb->configure (frame->yuv2rgb, frame->width, 16, 2*frame->frame.pitches[0], 2*frame->frame.pitches[1], frame->output_width, frame->stripe_height, frame->bytes_per_line*2); frame->yuv_stride = frame->bytes_per_line*2; break; case VO_BOTH_FIELDS: frame->yuv2rgb->configure (frame->yuv2rgb, frame->width, 16, frame->frame.pitches[0], frame->frame.pitches[1], frame->output_width, frame->stripe_height, frame->bytes_per_line); frame->yuv_stride = frame->bytes_per_line; break; } } } printf("after gui changed\n"); /* * reset dest pointers */ if (frame->data) { switch (flags) { case VO_TOP_FIELD: frame->rgb_dst = (uint8_t *)frame->data; frame->stripe_inc = 2 * frame->stripe_height * frame->bytes_per_line; break; case VO_BOTTOM_FIELD: frame->rgb_dst = (uint8_t *)frame->data + frame->bytes_per_line ; frame->stripe_inc = 2 * frame->stripe_height * frame->bytes_per_line; break; case VO_BOTH_FIELDS: frame->rgb_dst = (uint8_t *)frame->data; frame->stripe_inc = frame->stripe_height * frame->bytes_per_line; break; } } printf("done\n"); } static void null_display_frame( vo_driver_t* self, vo_frame_t *frame_gen ){ null_driver_t* this = (null_driver_t*) self; opie_frame_t* frame = (opie_frame_t*)frame_gen; display_xine_frame_t display = this->frameDis; printf("display frame\n"); // if( this->m_show_video ) { // return if not displaying printf("calling home aye\n" ); if( display != NULL ) { (*display)(this->caller, frame->data, frame->output_width, frame->output_height, frame->bytes_per_line ); printf("display done hope you enyoyed the frame"); } // } frame->frame.displayed (&frame->frame); } // blending related static void null_overlay_clut_yuv2rgb(null_driver_t *this, vo_overlay_t *overlay, opie_frame_t *frame) { int i; clut_t* clut = (clut_t*) overlay->color; if (!overlay->rgb_clut) { for (i = 0; i < sizeof(overlay->color)/sizeof(overlay->color[0]); i++) { *((uint32_t *)&clut[i]) = frame->yuv2rgb->yuv2rgb_single_pixel_fun (frame->yuv2rgb, clut[i].y, clut[i].cb, clut[i].cr); } overlay->rgb_clut++; } if (!overlay->clip_rgb_clut) { clut = (clut_t*) overlay->clip_color; for (i = 0; i < sizeof(overlay->color)/sizeof(overlay->color[0]); i++) { *((uint32_t *)&clut[i]) = frame->yuv2rgb->yuv2rgb_single_pixel_fun(frame->yuv2rgb, clut[i].y, clut[i].cb, clut[i].cr); } overlay->clip_rgb_clut++; } } static void null_overlay_blend (vo_driver_t *this_gen, vo_frame_t *frame_gen, vo_overlay_t *overlay) { null_driver_t *this = (null_driver_t *) this_gen; opie_frame_t *frame = (opie_frame_t *) frame_gen; printf("overlay blend\n"); if(!this->m_show_video || frame->output_width == 0 || frame->output_height== 0) return; /* Alpha Blend here */ if (overlay->rle) { if( !overlay->rgb_clut || !overlay->clip_rgb_clut) null_overlay_clut_yuv2rgb(this,overlay,frame); switch(this->bpp) { case 16: blend_rgb16( (uint8_t *)frame->data, overlay, frame->output_width, frame->output_height, frame->width, frame->height); break; case 24: blend_rgb24( (uint8_t *)frame->data, overlay, frame->output_width, frame->output_height, frame->width, frame->height); break; case 32: blend_rgb32( (uint8_t *)frame->data, overlay, frame->output_width, frame->output_height, frame->width, frame->height); break; default: /* It should never get here */ break; } } } static int null_get_property( vo_driver_t* self, int property ){ printf("property get\n"); return 0; } static int null_set_property( vo_driver_t* self, int property, int value ){ printf("set property\n"); return value; } static void null_get_property_min_max( vo_driver_t* self, int property, int *min, int *max ){ printf("min max\n"); *max = 0; *min = 0; } static int null_gui_data_exchange( vo_driver_t* self, int data_type, void *data ){ return 0; } static void null_exit( vo_driver_t* self ){ null_driver_t* this = (null_driver_t*)self; free ( this ); } static int null_redraw_needed( vo_driver_t* self ){ return 0; } vo_driver_t* init_video_out_plugin( config_values_t* conf, void* video ){ null_driver_t *vo; vo = (null_driver_t*)malloc( sizeof(null_driver_t ) ); /* memset? */ memset(vo,0, sizeof(null_driver_t ) ); vo->m_show_video = 0; // false vo->m_video_fullscreen = 0; vo->m_is_scaling = 0; vo->user_ratio = ASPECT_AUTO; vo->display_ratio = 1.0; vo->gui_width = 16; vo->gui_height = 8; vo->frameDis = NULL; /* install callback handlers*/ vo->vo_driver.get_capabilities = null_get_capabilities; vo->vo_driver.alloc_frame = null_alloc_frame; vo->vo_driver.update_frame_format = null_update_frame_format; vo->vo_driver.display_frame = null_display_frame; vo->vo_driver.overlay_blend = null_overlay_blend; vo->vo_driver.get_property = null_get_property; vo->vo_driver.set_property = null_set_property; vo->vo_driver.get_property_min_max = null_get_property_min_max; vo->vo_driver.gui_data_exchange = null_gui_data_exchange; vo->vo_driver.exit = null_exit; vo->vo_driver.redraw_needed = null_redraw_needed; /* capabilities */ vo->m_capabilities = VO_CAP_COPIES_IMAGE | VO_CAP_YUY2 | VO_CAP_YV12; vo->yuv2rgb_factory = yuv2rgb_factory_init (MODE_16_RGB, vo->yuv2rgb_swap, vo->yuv2rgb_cmap); printf("done initialisation\n"); return (vo_driver_t*) vo; } static vo_info_t vo_info_null = { 5, "null plugin", NULL, VISUAL_TYPE_FB, 5 }; vo_info_t *get_video_out_plugin_info(){ vo_info_null.description = _("xine video output plugin using null device"); return &vo_info_null; } /* this is special for this device */ /** * We know that we will be controled by the XINE LIB++ */ /** * */ int null_is_showing_video( vo_driver_t* self ){ null_driver_t* this = (null_driver_t*)self; return this->m_show_video; } void null_set_show_video( vo_driver_t* self, int show ) { ((null_driver_t*)self)->m_show_video = show; } int null_is_fullscreen( vo_driver_t* self ){ return ((null_driver_t*)self)->m_video_fullscreen; } void null_set_fullscreen( vo_driver_t* self, int screen ){ ((null_driver_t*)self)->m_video_fullscreen = screen; } int null_is_scaling( vo_driver_t* self ){ return ((null_driver_t*)self)->m_is_scaling; } void null_set_scaling( vo_driver_t* self, int scale ){ ((null_driver_t*)self)->m_is_scaling = scale; } void null_set_gui_width( vo_driver_t* self, int width ){ ((null_driver_t*)self)->gui_width = width; } void null_set_gui_height( vo_driver_t* self, int height ){ ((null_driver_t*)self)->gui_height = height; } void null_set_mode( vo_driver_t* self, int depth, int rgb ){ null_driver_t* this = (null_driver_t*)self; this->bytes_per_pixel = (depth + 7 ) / 8; this->bpp = this->bytes_per_pixel * 8; this->depth = depth; printf("depth %d %d\n", depth, this->bpp); printf("pixeltype %d\n", rgb ); switch ( this->depth ){ case 32: if( rgb == 0 ) this->yuv2rgb_mode = MODE_32_RGB; else this->yuv2rgb_mode = MODE_32_BGR; case 24: if( this->bpp == 32 ) { if(rgb == 0 ) this->yuv2rgb_mode = MODE_32_RGB; else this->yuv2rgb_mode = MODE_32_BGR; }else{ if( rgb == 0 ) this->yuv2rgb_mode = MODE_24_RGB; else this->yuv2rgb_mode = MODE_24_BGR; }; break; case 16: if( rgb == 0 ) this->yuv2rgb_mode = MODE_16_RGB; else this->yuv2rgb_mode = MODE_16_BGR; break; case 15: if( rgb == 0 ) this->yuv2rgb_mode = MODE_15_RGB; else this->yuv2rgb_mode = MODE_15_BGR; break; case 8: if( rgb == 0 ) this->yuv2rgb_mode = MODE_8_RGB; else this->yuv2rgb_mode = MODE_8_BGR; break; }; //free(this->yuv2rgb_factory ); // this->yuv2rgb_factory = yuv2rgb_factory_init (this->yuv2rgb_mode, this->yuv2rgb_swap, // this->yuv2rgb_cmap); }; void null_display_handler(vo_driver_t* self, display_xine_frame_t t, void* user_data) { null_driver_t* this = (null_driver_t*) self; this->caller = user_data; this->frameDis = t; } diff --git a/noncore/multimedia/opieplayer2/opieplayer2.pro b/noncore/multimedia/opieplayer2/opieplayer2.pro index d8cacd0..fee9242 100644 --- a/noncore/multimedia/opieplayer2/opieplayer2.pro +++ b/noncore/multimedia/opieplayer2/opieplayer2.pro @@ -1,23 +1,23 @@ TEMPLATE = app CONFIG = qt warn_on release #release DESTDIR = $(OPIEDIR)/bin HEADERS = playlistselection.h mediaplayerstate.h xinecontrol.h mediadetect.h\ videowidget.h audiowidget.h playlistwidget.h mediaplayer.h inputDialog.h \ frame.h lib.h xinevideowidget.h \ alphablend.h yuv2rgb.h SOURCES = main.cpp \ playlistselection.cpp mediaplayerstate.cpp xinecontrol.cpp mediadetect.cpp\ videowidget.cpp audiowidget.cpp playlistwidget.cpp mediaplayer.cpp inputDialog.cpp \ frame.cpp lib.cpp nullvideo.c xinevideowidget.cpp \ - alphablend.c yuv2rgb.c yuv2rgb_mlib.c yuv2rgb_mmx.c + alphablend.c yuv2rgb.c yuv2rgb_arm.c yuv2rgb_arm4l.S TARGET = opieplayer2 INCLUDEPATH += $(OPIEDIR)/include DEPENDPATH += $(OPIEDIR)/include LIBS += -lqpe -lpthread -lopie -lxine -lxineutils MOC_DIR=qpeobj OBJECTS_DIR=qpeobj INCLUDEPATH += $(OPIEDIR)/include DEPENDPATH += $(OPIEDIR)/include diff --git a/noncore/multimedia/opieplayer2/yuv2rgb.c b/noncore/multimedia/opieplayer2/yuv2rgb.c index d1d6627..22bb4cb 100644 --- a/noncore/multimedia/opieplayer2/yuv2rgb.c +++ b/noncore/multimedia/opieplayer2/yuv2rgb.c @@ -2116,1045 +2116,1053 @@ static void yuv2rgb_c_palette (yuv2rgb_t *this, uint8_t * _dst, dst_2 = _dst + this->rgb_stride; py_1 = _py; py_2 = _py + this->y_stride; pu = _pu; pv = _pv; width = this->source_width >> 3; do { RGB(0); DST1CMAP(0); DST2CMAP(0); RGB(1); DST2CMAP(1); DST1CMAP(1); RGB(2); DST1CMAP(2); DST2CMAP(2); RGB(3); DST2CMAP(3); DST1CMAP(3); pu += 4; pv += 4; py_1 += 8; py_2 += 8; dst_1 += 8; dst_2 += 8; } while (--width); _dst += 2 * this->rgb_stride; _py += 2 * this->y_stride; _pu += this->uv_stride; _pv += this->uv_stride; } while (--height); } } static int div_round (int dividend, int divisor) { if (dividend > 0) return (dividend + (divisor>>1)) / divisor; else return -((-dividend + (divisor>>1)) / divisor); } static void yuv2rgb_setup_tables (yuv2rgb_factory_t *this, int mode, int swapped) { int i; uint8_t table_Y[1024]; uint32_t * table_32 = 0; uint16_t * table_16 = 0; uint8_t * table_8 = 0; int entry_size = 0; void *table_r = 0, *table_g = 0, *table_b = 0; int shift_r = 0, shift_g = 0, shift_b = 0; int crv = Inverse_Table_6_9[this->matrix_coefficients][0]; int cbu = Inverse_Table_6_9[this->matrix_coefficients][1]; int cgu = -Inverse_Table_6_9[this->matrix_coefficients][2]; int cgv = -Inverse_Table_6_9[this->matrix_coefficients][3]; for (i = 0; i < 1024; i++) { int j; j = (76309 * (i - 384 - 16) + 32768) >> 16; j = (j < 0) ? 0 : ((j > 255) ? 255 : j); table_Y[i] = j; } switch (mode) { case MODE_32_RGB: case MODE_32_BGR: table_32 = malloc ((197 + 2*682 + 256 + 132) * sizeof (uint32_t)); entry_size = sizeof (uint32_t); table_r = table_32 + 197; table_b = table_32 + 197 + 685; table_g = table_32 + 197 + 2*682; if (swapped) { switch (mode) { case MODE_32_RGB: shift_r = 8; shift_g = 16; shift_b = 24; break; case MODE_32_BGR: shift_r = 24; shift_g = 16; shift_b = 8; break; } } else { switch (mode) { case MODE_32_RGB: shift_r = 16; shift_g = 8; shift_b = 0; break; case MODE_32_BGR: shift_r = 0; shift_g = 8; shift_b = 16; break; } } for (i = -197; i < 256+197; i++) ((uint32_t *) table_r)[i] = table_Y[i+384] << shift_r; for (i = -132; i < 256+132; i++) ((uint32_t *) table_g)[i] = table_Y[i+384] << shift_g; for (i = -232; i < 256+232; i++) ((uint32_t *) table_b)[i] = table_Y[i+384] << shift_b; break; case MODE_24_RGB: case MODE_24_BGR: table_8 = malloc ((256 + 2*232) * sizeof (uint8_t)); entry_size = sizeof (uint8_t); table_r = table_g = table_b = table_8 + 232; for (i = -232; i < 256+232; i++) ((uint8_t * )table_b)[i] = table_Y[i+384]; break; case MODE_15_BGR: case MODE_16_BGR: case MODE_15_RGB: case MODE_16_RGB: table_16 = malloc ((197 + 2*682 + 256 + 132) * sizeof (uint16_t)); entry_size = sizeof (uint16_t); table_r = table_16 + 197; table_b = table_16 + 197 + 685; table_g = table_16 + 197 + 2*682; if (swapped) { switch (mode) { case MODE_15_BGR: shift_r = 8; shift_g = 5; shift_b = 2; break; case MODE_16_BGR: shift_r = 8; shift_g = 5; shift_b = 3; break; case MODE_15_RGB: shift_r = 2; shift_g = 5; shift_b = 8; break; case MODE_16_RGB: shift_r = 3; shift_g = 5; shift_b = 8; break; } } else { switch (mode) { case MODE_15_BGR: shift_r = 0; shift_g = 5; shift_b = 10; break; case MODE_16_BGR: shift_r = 0; shift_g = 5; shift_b = 11; break; case MODE_15_RGB: shift_r = 10; shift_g = 5; shift_b = 0; break; case MODE_16_RGB: shift_r = 11; shift_g = 5; shift_b = 0; break; } } for (i = -197; i < 256+197; i++) ((uint16_t *)table_r)[i] = (table_Y[i+384] >> 3) << shift_r; for (i = -132; i < 256+132; i++) { int j = table_Y[i+384] >> (((mode==MODE_16_RGB) || (mode==MODE_16_BGR)) ? 2 : 3); if (swapped) ((uint16_t *)table_g)[i] = (j&7) << 13 | (j>>3); else ((uint16_t *)table_g)[i] = j << 5; } for (i = -232; i < 256+232; i++) ((uint16_t *)table_b)[i] = (table_Y[i+384] >> 3) << shift_b; break; case MODE_8_RGB: case MODE_8_BGR: table_8 = malloc ((197 + 2*682 + 256 + 132) * sizeof (uint8_t)); entry_size = sizeof (uint8_t); table_r = table_8 + 197; table_b = table_8 + 197 + 685; table_g = table_8 + 197 + 2*682; switch (mode) { case MODE_8_RGB: shift_r = 5; shift_g = 2; shift_b = 0; break; case MODE_8_BGR: shift_r = 0; shift_g = 3; shift_b = 6; break; } for (i = -197; i < 256+197; i++) ((uint8_t *) table_r)[i] = (table_Y[i+384] >> 5) << shift_r; for (i = -132; i < 256+132; i++) ((uint8_t *) table_g)[i] = (table_Y[i+384] >> 5) << shift_g; for (i = -232; i < 256+232; i++) ((uint8_t *) table_b)[i] = (table_Y[i+384] >> 6) << shift_b; break; case MODE_8_GRAY: return; case MODE_PALETTE: table_16 = malloc ((197 + 2*682 + 256 + 132) * sizeof (uint16_t)); entry_size = sizeof (uint16_t); table_r = table_16 + 197; table_b = table_16 + 197 + 685; table_g = table_16 + 197 + 2*682; shift_r = 10; shift_g = 5; shift_b = 0; for (i = -197; i < 256+197; i++) ((uint16_t *)table_r)[i] = (table_Y[i+384] >> 3) << 10; for (i = -132; i < 256+132; i++) ((uint16_t *)table_g)[i] = (table_Y[i+384] >> 3) << 5; for (i = -232; i < 256+232; i++) ((uint16_t *)table_b)[i] = (table_Y[i+384] >> 3) << 0; break; default: fprintf (stderr, "mode %d not supported by yuv2rgb\n", mode); abort(); } for (i = 0; i < 256; i++) { this->table_rV[i] = (((uint8_t *) table_r) + entry_size * div_round (crv * (i-128), 76309)); this->table_gU[i] = (((uint8_t *) table_g) + entry_size * div_round (cgu * (i-128), 76309)); this->table_gV[i] = entry_size * div_round (cgv * (i-128), 76309); this->table_bU[i] = (((uint8_t *)table_b) + entry_size * div_round (cbu * (i-128), 76309)); } this->gamma = 0; this->entry_size = entry_size; } static uint32_t yuv2rgb_single_pixel_32 (yuv2rgb_t *this, uint8_t y, uint8_t u, uint8_t v) { uint32_t * r, * g, * b; r = this->table_rV[v]; g = (void *) (((uint8_t *)this->table_gU[u]) + this->table_gV[v]); b = this->table_bU[u]; return r[y] + g[y] + b[y]; } static uint32_t yuv2rgb_single_pixel_24_rgb (yuv2rgb_t *this, uint8_t y, uint8_t u, uint8_t v) { uint8_t * r, * g, * b; r = this->table_rV[v]; g = (void *) (((uint8_t *)this->table_gU[u]) + this->table_gV[v]); b = this->table_bU[u]; return (uint32_t) r[y] + ((uint32_t) g[y] << 8) + ((uint32_t) b[y] << 16); } static uint32_t yuv2rgb_single_pixel_24_bgr (yuv2rgb_t *this, uint8_t y, uint8_t u, uint8_t v) { uint8_t * r, * g, * b; r = this->table_rV[v]; g = (void *) (((uint8_t *)this->table_gU[u]) + this->table_gV[v]); b = this->table_bU[u]; return (uint32_t) b[y] + ((uint32_t) g[y] << 8) + ((uint32_t) r[y] << 16); } static uint32_t yuv2rgb_single_pixel_16 (yuv2rgb_t *this, uint8_t y, uint8_t u, uint8_t v) { uint16_t * r, * g, * b; r = this->table_rV[v]; g = (void *) (((uint8_t *)this->table_gU[u]) + this->table_gV[v]); b = this->table_bU[u]; return r[y] + g[y] + b[y]; } static uint32_t yuv2rgb_single_pixel_8 (yuv2rgb_t *this, uint8_t y, uint8_t u, uint8_t v) { uint8_t * r, * g, * b; r = this->table_rV[v]; g = (void *) (((uint8_t *)this->table_gU[u]) + this->table_gV[v]); b = this->table_bU[u]; return r[y] + g[y] + b[y]; } static uint32_t yuv2rgb_single_pixel_gray (yuv2rgb_t *this, uint8_t y, uint8_t u, uint8_t v) { return y; } static uint32_t yuv2rgb_single_pixel_palette (yuv2rgb_t *this, uint8_t y, uint8_t u, uint8_t v) { uint16_t * r, * g, * b; r = this->table_rV[v]; g = (void *) (((uint8_t *)this->table_gU[u]) + this->table_gV[v]); b = this->table_bU[u]; return this->cmap[r[y] + g[y] + b[y]]; } static void yuv2rgb_c_init (yuv2rgb_factory_t *this) { switch (this->mode) { case MODE_32_RGB: case MODE_32_BGR: this->yuv2rgb_fun = yuv2rgb_c_32; break; case MODE_24_RGB: case MODE_24_BGR: this->yuv2rgb_fun = (this->mode==MODE_24_RGB && !this->swapped) || (this->mode==MODE_24_BGR && this->swapped) ? yuv2rgb_c_24_rgb : yuv2rgb_c_24_bgr; break; case MODE_15_BGR: case MODE_16_BGR: case MODE_15_RGB: case MODE_16_RGB: this->yuv2rgb_fun = yuv2rgb_c_16; break; case MODE_8_RGB: case MODE_8_BGR: this->yuv2rgb_fun = yuv2rgb_c_8; break; case MODE_8_GRAY: this->yuv2rgb_fun = yuv2rgb_c_gray; break; case MODE_PALETTE: this->yuv2rgb_fun = yuv2rgb_c_palette; break; default: printf ("yuv2rgb: mode %d not supported by yuv2rgb\n", this->mode); abort(); } } static void yuv2rgb_single_pixel_init (yuv2rgb_factory_t *this) { switch (this->mode) { case MODE_32_RGB: case MODE_32_BGR: this->yuv2rgb_single_pixel_fun = yuv2rgb_single_pixel_32; break; case MODE_24_RGB: case MODE_24_BGR: this->yuv2rgb_single_pixel_fun = (this->mode==MODE_24_RGB && !this->swapped) || (this->mode==MODE_24_BGR && this->swapped) ? yuv2rgb_single_pixel_24_rgb : yuv2rgb_single_pixel_24_bgr; break; case MODE_15_BGR: case MODE_16_BGR: case MODE_15_RGB: case MODE_16_RGB: this->yuv2rgb_single_pixel_fun = yuv2rgb_single_pixel_16; break; case MODE_8_RGB: case MODE_8_BGR: this->yuv2rgb_single_pixel_fun = yuv2rgb_single_pixel_8; break; case MODE_8_GRAY: this->yuv2rgb_single_pixel_fun = yuv2rgb_single_pixel_gray; break; case MODE_PALETTE: this->yuv2rgb_single_pixel_fun = yuv2rgb_single_pixel_palette; break; default: printf ("yuv2rgb: mode %d not supported by yuv2rgb\n", this->mode); abort(); } } /* * yuy2 stuff */ static void yuy22rgb_c_32 (yuv2rgb_t *this, uint8_t * _dst, uint8_t * _p) { int U, V, Y; uint8_t * py_1, * pu, * pv; uint32_t * r, * g, * b; uint32_t * dst_1; int width, height; int dy; /* FIXME: implement unscaled version */ scale_line_4 (_p+1, this->u_buffer, this->dest_width >> 1, this->step_dx); scale_line_4 (_p+3, this->v_buffer, this->dest_width >> 1, this->step_dx); scale_line_2 (_p, this->y_buffer, this->dest_width, this->step_dx); dy = 0; height = this->dest_height; for (;;) { dst_1 = (uint32_t*)_dst; py_1 = this->y_buffer; pu = this->u_buffer; pv = this->v_buffer; width = this->dest_width >> 3; do { RGB(0); DST1(0); RGB(1); DST1(1); RGB(2); DST1(2); RGB(3); DST1(3); pu += 4; pv += 4; py_1 += 8; dst_1 += 8; } while (--width); dy += this->step_dy; _dst += this->rgb_stride; while (--height > 0 && dy < 32768) { xine_fast_memcpy (_dst, (uint8_t*)_dst-this->rgb_stride, this->dest_width*4); dy += this->step_dy; _dst += this->rgb_stride; } if (height <= 0) break; _p += this->y_stride*2*(dy>>15); dy &= 32767; /* dy -= 32768; _p += this->y_stride*2; */ scale_line_4 (_p+1, this->u_buffer, this->dest_width >> 1, this->step_dx); scale_line_4 (_p+3, this->v_buffer, this->dest_width >> 1, this->step_dx); scale_line_2 (_p, this->y_buffer, this->dest_width, this->step_dx); } } static void yuy22rgb_c_24_rgb (yuv2rgb_t *this, uint8_t * _dst, uint8_t * _p) { int U, V, Y; uint8_t * py_1, * pu, * pv; uint8_t * r, * g, * b; uint8_t * dst_1; int width, height; int dy; /* FIXME: implement unscaled version */ scale_line_4 (_p+1, this->u_buffer, this->dest_width >> 1, this->step_dx); scale_line_4 (_p+3, this->v_buffer, this->dest_width >> 1, this->step_dx); scale_line_2 (_p, this->y_buffer, this->dest_width, this->step_dx); dy = 0; height = this->dest_height; for (;;) { dst_1 = _dst; py_1 = this->y_buffer; pu = this->u_buffer; pv = this->v_buffer; width = this->dest_width >> 3; do { RGB(0); DST1RGB(0); RGB(1); DST1RGB(1); RGB(2); DST1RGB(2); RGB(3); DST1RGB(3); pu += 4; pv += 4; py_1 += 8; dst_1 += 24; } while (--width); dy += this->step_dy; _dst += this->rgb_stride; while (--height > 0 && dy < 32768) { xine_fast_memcpy (_dst, (uint8_t*)_dst-this->rgb_stride, this->dest_width*3); dy += this->step_dy; _dst += this->rgb_stride; } if (height <= 0) break; _p += this->y_stride*2*(dy>>15); dy &= 32767; /* dy -= 32768; _p += this->y_stride*2; */ scale_line_4 (_p+1, this->u_buffer, this->dest_width >> 1, this->step_dx); scale_line_4 (_p+3, this->v_buffer, this->dest_width >> 1, this->step_dx); scale_line_2 (_p, this->y_buffer, this->dest_width, this->step_dx); } } static void yuy22rgb_c_24_bgr (yuv2rgb_t *this, uint8_t * _dst, uint8_t * _p) { int U, V, Y; uint8_t * py_1, * pu, * pv; uint8_t * r, * g, * b; uint8_t * dst_1; int width, height; int dy; /* FIXME: implement unscaled version */ scale_line_4 (_p+1, this->u_buffer, this->dest_width >> 1, this->step_dx); scale_line_4 (_p+3, this->v_buffer, this->dest_width >> 1, this->step_dx); scale_line_2 (_p, this->y_buffer, this->dest_width, this->step_dx); dy = 0; height = this->dest_height; for (;;) { dst_1 = _dst; py_1 = this->y_buffer; pu = this->u_buffer; pv = this->v_buffer; width = this->dest_width >> 3; do { RGB(0); DST1BGR(0); RGB(1); DST1BGR(1); RGB(2); DST1BGR(2); RGB(3); DST1BGR(3); pu += 4; pv += 4; py_1 += 8; dst_1 += 24; } while (--width); dy += this->step_dy; _dst += this->rgb_stride; while (--height > 0 && dy < 32768) { xine_fast_memcpy (_dst, (uint8_t*)_dst-this->rgb_stride, this->dest_width*3); dy += this->step_dy; _dst += this->rgb_stride; } if (height <= 0) break; _p += this->y_stride*2*(dy>>15); dy &= 32767; scale_line_4 (_p+1, this->u_buffer, this->dest_width >> 1, this->step_dx); scale_line_4 (_p+3, this->v_buffer, this->dest_width >> 1, this->step_dx); scale_line_2 (_p, this->y_buffer, this->dest_width, this->step_dx); } } static void yuy22rgb_c_16 (yuv2rgb_t *this, uint8_t * _dst, uint8_t * _p) { int U, V, Y; uint8_t * py_1, * pu, * pv; uint16_t * r, * g, * b; uint16_t * dst_1; int width, height; int dy; /* FIXME: implement unscaled version */ scale_line_4 (_p+1, this->u_buffer, this->dest_width >> 1, this->step_dx); scale_line_4 (_p+3, this->v_buffer, this->dest_width >> 1, this->step_dx); scale_line_2 (_p, this->y_buffer, this->dest_width, this->step_dx); dy = 0; height = this->dest_height; for (;;) { dst_1 = (uint16_t*)_dst; py_1 = this->y_buffer; pu = this->u_buffer; pv = this->v_buffer; width = this->dest_width >> 3; do { RGB(0); DST1(0); RGB(1); DST1(1); RGB(2); DST1(2); RGB(3); DST1(3); pu += 4; pv += 4; py_1 += 8; dst_1 += 8; } while (--width); dy += this->step_dy; _dst += this->rgb_stride; while (--height > 0 && dy < 32768) { xine_fast_memcpy (_dst, (uint8_t*)_dst-this->rgb_stride, this->dest_width*2); dy += this->step_dy; _dst += this->rgb_stride; } if (height <= 0) break; _p += this->y_stride*2*(dy>>15); dy &= 32767; scale_line_4 (_p+1, this->u_buffer, this->dest_width >> 1, this->step_dx); scale_line_4 (_p+3, this->v_buffer, this->dest_width >> 1, this->step_dx); scale_line_2 (_p, this->y_buffer, this->dest_width, this->step_dx); } } static void yuy22rgb_c_8 (yuv2rgb_t *this, uint8_t * _dst, uint8_t * _p) { int U, V, Y; uint8_t * py_1, * pu, * pv; uint8_t * r, * g, * b; uint8_t * dst_1; int width, height; int dy; /* FIXME: implement unscaled version */ scale_line_4 (_p+1, this->u_buffer, this->dest_width >> 1, this->step_dx); scale_line_4 (_p+3, this->v_buffer, this->dest_width >> 1, this->step_dx); scale_line_2 (_p, this->y_buffer, this->dest_width, this->step_dx); dy = 0; height = this->dest_height; for (;;) { dst_1 = _dst; py_1 = this->y_buffer; pu = this->u_buffer; pv = this->v_buffer; width = this->dest_width >> 3; do { RGB(0); DST1(0); RGB(1); DST1(1); RGB(2); DST1(2); RGB(3); DST1(3); pu += 4; pv += 4; py_1 += 8; dst_1 += 8; } while (--width); dy += this->step_dy; _dst += this->rgb_stride; while (--height > 0 && dy < 32768) { xine_fast_memcpy (_dst, (uint8_t*)_dst-this->rgb_stride, this->dest_width); dy += this->step_dy; _dst += this->rgb_stride; } if (height <= 0) break; _p += this->y_stride*2*(dy>>15); dy &= 32767; scale_line_4 (_p+1, this->u_buffer, this->dest_width >> 1, this->step_dx); scale_line_4 (_p+3, this->v_buffer, this->dest_width >> 1, this->step_dx); scale_line_2 (_p, this->y_buffer, this->dest_width, this->step_dx); } } static void yuy22rgb_c_gray (yuv2rgb_t *this, uint8_t * _dst, uint8_t * _p) { int width, height; int dy; uint8_t * dst; uint8_t * y; if (this->do_scale) { dy = 0; height = this->dest_height; for (;;) { scale_line_2 (_p, _dst, this->dest_width, this->step_dx); dy += this->step_dy; _dst += this->rgb_stride; while (--height > 0 && dy < 32768) { xine_fast_memcpy (_dst, (uint8_t*)_dst-this->rgb_stride, this->dest_width); dy += this->step_dy; _dst += this->rgb_stride; } if (height <= 0) break; _p += this->y_stride*2*(dy>>15); dy &= 32767; } } else { for (height = this->source_height; --height >= 0; ) { dst = _dst; y = _p; for (width = this->source_width; --width >= 0; ) { *dst++ = *y; y += 2; } _dst += this->rgb_stride; _p += this->y_stride*2; } } } static void yuy22rgb_c_palette (yuv2rgb_t *this, uint8_t * _dst, uint8_t * _p) { int U, V, Y; uint8_t * py_1, * pu, * pv; uint16_t * r, * g, * b; uint8_t * dst_1; int width, height; int dy; scale_line_4 (_p+1, this->u_buffer, this->dest_width >> 1, this->step_dx); scale_line_4 (_p+3, this->v_buffer, this->dest_width >> 1, this->step_dx); scale_line_2 (_p, this->y_buffer, this->dest_width, this->step_dx); dy = 0; height = this->dest_height; for (;;) { dst_1 = _dst; py_1 = this->y_buffer; pu = this->u_buffer; pv = this->v_buffer; width = this->dest_width >> 3; do { RGB(0); DST1CMAP(0); RGB(1); DST1CMAP(1); RGB(2); DST1CMAP(2); RGB(3); DST1CMAP(3); pu += 4; pv += 4; py_1 += 8; dst_1 += 8; } while (--width); dy += this->step_dy; _dst += this->rgb_stride; while (--height > 0 && dy < 32768) { xine_fast_memcpy (_dst, (uint8_t*)_dst-this->rgb_stride, this->dest_width); dy += this->step_dy; _dst += this->rgb_stride; } if (height <= 0) break; _p += this->y_stride*2*(dy>>15); dy &= 32767; scale_line_4 (_p+1, this->u_buffer, this->dest_width >> 1, this->step_dx); scale_line_4 (_p+3, this->v_buffer, this->dest_width >> 1, this->step_dx); scale_line_2 (_p, this->y_buffer, this->dest_width, this->step_dx); } } static void yuy22rgb_c_init (yuv2rgb_factory_t *this) { switch (this->mode) { case MODE_32_RGB: case MODE_32_BGR: this->yuy22rgb_fun = yuy22rgb_c_32; break; case MODE_24_RGB: case MODE_24_BGR: this->yuy22rgb_fun = (this->mode==MODE_24_RGB && !this->swapped) || (this->mode==MODE_24_BGR && this->swapped) ? yuy22rgb_c_24_rgb : yuy22rgb_c_24_bgr; break; case MODE_15_BGR: case MODE_16_BGR: case MODE_15_RGB: case MODE_16_RGB: this->yuy22rgb_fun = yuy22rgb_c_16; break; case MODE_8_RGB: case MODE_8_BGR: this->yuy22rgb_fun = yuy22rgb_c_8; break; case MODE_8_GRAY: this->yuy22rgb_fun = yuy22rgb_c_gray; break; case MODE_PALETTE: this->yuy22rgb_fun = yuy22rgb_c_palette; break; default: printf ("yuv2rgb: mode %d not supported for yuy2\n", this->mode); } } yuv2rgb_t *yuv2rgb_create_converter (yuv2rgb_factory_t *factory) { yuv2rgb_t *this = xine_xmalloc (sizeof (yuv2rgb_t)); this->cmap = factory->cmap; this->y_chunk = this->y_buffer = NULL; this->u_chunk = this->u_buffer = NULL; this->v_chunk = this->v_buffer = NULL; this->table_rV = factory->table_rV; this->table_gU = factory->table_gU; this->table_gV = factory->table_gV; this->table_bU = factory->table_bU; this->yuv2rgb_fun = factory->yuv2rgb_fun; this->yuy22rgb_fun = factory->yuy22rgb_fun; this->yuv2rgb_single_pixel_fun = factory->yuv2rgb_single_pixel_fun; this->configure = yuv2rgb_configure; return this; } /* * factory functions */ void yuv2rgb_set_gamma (yuv2rgb_factory_t *this, int gamma) { int i; for (i = 0; i < 256; i++) { (uint8_t *)this->table_rV[i] += this->entry_size*(gamma - this->gamma); (uint8_t *)this->table_gU[i] += this->entry_size*(gamma - this->gamma); (uint8_t *)this->table_bU[i] += this->entry_size*(gamma - this->gamma); } #ifdef ARCH_X86 mmx_yuv2rgb_set_gamma(gamma); #endif this->gamma = gamma; } int yuv2rgb_get_gamma (yuv2rgb_factory_t *this) { return this->gamma; } yuv2rgb_factory_t* yuv2rgb_factory_init (int mode, int swapped, uint8_t *cmap) { yuv2rgb_factory_t *this; #ifdef ARCH_X86 uint32_t mm = xine_mm_accel(); #endif this = malloc (sizeof (yuv2rgb_factory_t)); this->mode = mode; this->swapped = swapped; this->cmap = cmap; this->create_converter = yuv2rgb_create_converter; this->set_gamma = yuv2rgb_set_gamma; this->get_gamma = yuv2rgb_get_gamma; this->matrix_coefficients = 6; yuv2rgb_setup_tables (this, mode, swapped); /* * auto-probe for the best yuv2rgb function */ this->yuv2rgb_fun = NULL; #ifdef ARCH_X86 if ((this->yuv2rgb_fun == NULL) && (mm & MM_ACCEL_X86_MMXEXT)) { yuv2rgb_init_mmxext (this); if (this->yuv2rgb_fun != NULL) printf ("yuv2rgb: using MMXEXT for colorspace transform\n"); } if ((this->yuv2rgb_fun == NULL) && (mm & MM_ACCEL_X86_MMX)) { yuv2rgb_init_mmx (this); if (this->yuv2rgb_fun != NULL) printf ("yuv2rgb: using MMX for colorspace transform\n"); } #endif #if HAVE_MLIB if (this->yuv2rgb_fun == NULL) { yuv2rgb_init_mlib (this); if (this->yuv2rgb_fun != NULL) printf ("yuv2rgb: using medialib for colorspace transform\n"); } #endif +#ifdef __arm__ + if (this->yuv2rgb_fun == NULL) { + yuv2rgb_init_arm ( this ); + + if(this->yuv2rgb_fun != NULL) + printf("yuv2rgb: using arm4l assembler for colorspace transform\n" ); + } +#endif if (this->yuv2rgb_fun == NULL) { printf ("yuv2rgb: no accelerated colorspace conversion found\n"); yuv2rgb_c_init (this); } /* * auto-probe for the best yuy22rgb function */ /* FIXME: implement mmx/mlib functions */ yuy22rgb_c_init (this); /* * set up single pixel function */ yuv2rgb_single_pixel_init (this); return this; } diff --git a/noncore/multimedia/opieplayer2/yuv2rgb_arm.c b/noncore/multimedia/opieplayer2/yuv2rgb_arm.c new file mode 100644 index 0000000..699ee48 --- a/dev/null +++ b/noncore/multimedia/opieplayer2/yuv2rgb_arm.c @@ -0,0 +1,174 @@ +/* + * yuv2rgb_arm.c + * Copyright (C) 2000-2001 Project OPIE. + * All Rights Reserved. + * + * Author: Robert Griebl <sandman@handhelds.org> + * + * This file is part of OpiePlayer2. + * + * OpiePlayer2 is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + * + * OpiePlayer2 is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + */ + +#ifdef __arm__ + +#include <stdio.h> +#include <stdlib.h> +#include <string.h> +#include <inttypes.h> + +#include "yuv2rgb.h" +#include <xine/xineutils.h> + +#define RGB(i) \ + U = pu[i]; \ + V = pv[i]; \ + r = this->table_rV[V]; \ + g = (void *) (((uint8_t *)this->table_gU[U]) + this->table_gV[V]); \ + b = this->table_bU[U]; + +#define DST1(i) \ + Y = py_1[2*i]; \ + dst_1[2*i] = r[Y] + g[Y] + b[Y]; \ + Y = py_1[2*i+1]; \ + dst_1[2*i+1] = r[Y] + g[Y] + b[Y]; + + +struct dummy { + uint8_t *yuv [3]; + int stride [3]; +}; + +extern void convert_yuv420_rgb565(struct dummy *picture, unsigned char *results, int w, int h) ; + + +static void arm_rgb16 (yuv2rgb_t *this, uint8_t * _dst, + uint8_t * _py, uint8_t * _pu, uint8_t * _pv) +{ + if ( !this-> do_scale ) { + struct dummy d; + d. yuv [0] = _py; + d. yuv [1] = _pu; + d. yuv [2] = _pv; + d. stride [0] = this-> y_stride; + d. stride [1] = d. stride [2] = this-> uv_stride; + +// printf( "calling arm (%dx%d)\n", this-> dest_width, this-> dest_height ); + + convert_yuv420_rgb565 ( &d, _dst, this->dest_width, this->dest_height ); + +// printf ( "arm done\n" ); + } + else { + int U, V, Y; + uint8_t * py_1, * py_2, * pu, * pv; + uint16_t * r, * g, * b; + uint16_t * dst_1, * dst_2; + int width, height, dst_height; + int dy; + + scale_line_func_t scale_line = this->scale_line; + + scale_line (_pu, this->u_buffer, + this->dest_width >> 1, this->step_dx); + scale_line (_pv, this->v_buffer, + this->dest_width >> 1, this->step_dx); + scale_line (_py, this->y_buffer, + this->dest_width, this->step_dx); + + dy = 0; + dst_height = this->dest_height; + + for (height = 0;; ) { + dst_1 = (uint16_t*)_dst; + py_1 = this->y_buffer; + pu = this->u_buffer; + pv = this->v_buffer; + + width = this->dest_width >> 3; + + do { + RGB(0); + DST1(0); + + RGB(1); + DST1(1); + + RGB(2); + DST1(2); + + RGB(3); + DST1(3); + + pu += 4; + pv += 4; + py_1 += 8; + dst_1 += 8; + } while (--width); + + dy += this->step_dy; + _dst += this->rgb_stride; + + while (--dst_height > 0 && dy < 32768) { + + xine_fast_memcpy (_dst, (uint8_t*)_dst-this->rgb_stride, this->dest_width*2); + + dy += this->step_dy; + _dst += this->rgb_stride; + } + + if (dst_height <= 0) + break; + + do { + dy -= 32768; + _py += this->y_stride; + + scale_line (_py, this->y_buffer, + this->dest_width, this->step_dx); + + if (height & 1) { + _pu += this->uv_stride; + _pv += this->uv_stride; + + scale_line (_pu, this->u_buffer, + this->dest_width >> 1, this->step_dx); + scale_line (_pv, this->v_buffer, + this->dest_width >> 1, this->step_dx); + + } + height++; + } while( dy>=32768); + } + } +} + + + +void yuv2rgb_init_arm (yuv2rgb_factory_t *this) { + + if (this->swapped) + return; /*no swapped pixel output upto now*/ + + switch (this->mode) { + case MODE_16_RGB: + this->yuv2rgb_fun = arm_rgb16; + break; + } +} + + + +#endif diff --git a/noncore/multimedia/opieplayer2/yuv2rgb_arm4l.S b/noncore/multimedia/opieplayer2/yuv2rgb_arm4l.S new file mode 100644 index 0000000..f4a3395 --- a/dev/null +++ b/noncore/multimedia/opieplayer2/yuv2rgb_arm4l.S @@ -0,0 +1,192 @@ +/* WARNING : this function only works when stride_U == stride_V (I use some hacks to + not have to do too many computations at line's end)... + + C-like prototype : + void convert_yuv420_rgb565(AVPicture *picture, unsigned char *results, int w, int h) ; + +*/ + +#ifdef __arm__ + + .text + .align + + .global convert_yuv420_rgb565 +convert_yuv420_rgb565: + stmdb sp!, { r4 - r12, lr } @ all callee saved regs + ldr r7, [r0, #0] @ Y ptr + ldr r9, [r0, #4] @ U ptr + ldr r10, [r0, #8] @ V ptr + subs r10, r10, r9 @ V ptr - U ptr + ldr r8, [r0, #12] + add r8, r8, r7 @ Y + stride_Y + ldr r4, [r0, #12] @ Stride_Y + mov r4, r4, lsl #1 + sub r4, r4, r2 @ (2 * Stride_Y) - width + ldr r5, [r0, #16] @ Stride_U + sub r5, r5, r2, lsr #1 @ Stride_U - (width / 2) + ldr r6, [r0, #20] @ Stride_V + sub r6, r6, r2, lsr #1 @ Stride_V - (width / 2) + add r0, r1, r2, lsl #1 @ RGB + 1 + stmdb sp!, { r0-r10 } + @ Stack description : + @ (sp+ 0) RGB + one line + @ (sp+ 4) RGB + @ (sp+ 8) width (save) + @ (sp+12) height + @ (sp+16) (2 * stride_Y) - width + @ (sp+20) stride_U - (width / 2) + @ (sp+24) stride_V - (width / 2) !!! UNUSED !!! + @ (sp+28) Y ptr + @ (sp+32) Y ptr + one line + @ (sp+36) U ptr + @ (sp+40) V - U + mov lr, r2 @ Initialize the width counter + add r0, pc, #(const_storage-.-8) @ r0 = base pointer to the constants array + ldr r8, [r0, #(4*4)] @ r8 = multy +yuv_loop: + add r0, pc, #(const_storage-.-8) @ r0 = base pointer to the constants array + ldr r10, [sp, #28] @ r10 = Y + ldr r1, [sp, #36] @ r1 = U + ldrb r9, [r10, #0] @ r9 = *Y + ldrb r11, [r1] @ r11 = *U + add r1, r1, #1 @ r1 = U++ + ldr r2, [sp, #40] @ r2 = V - U + str r1, [sp, #36] @ store U++ + add r2, r1, r2 @ r2 = V+1 + ldrb r12, [r2, #-1] @ r12 = *V + sub r11, r11, #128 @ r11 = *U - 128 + sub r12, r12, #128 @ r12 = *V - 128 + ldr r1, [r0, #(4*0)] @ r1 = crv + mov r7, #32768 @ r7 = 32768 (for additions in MLA) + ldr r2, [r0, #(4*3)] @ r2 = -cgv + mla r6, r1, r12, r7 @ r6 = nonyc_r = crv * (*V - 128) + 32768 + ldr r3, [r0, #(4*1)] @ r3 = cbu + mla r4, r2, r12, r7 @ r4 = - cgv * (*V - 128) + 32768 + sub r9, r9, #16 @ r9 = *Y - 16 + mla r5, r3, r11, r7 @ r5 = nonyc_b = cbu * (*U - 128) + 32768 + ldr r0, [r0, #(4*2)] @ r0 = -cgu + mla r7, r8, r9, r6 @ r7 = (*Y - 16) * multy + nonyc_r + add r10, r10, #2 @ r10 = Y + 2 + mla r4, r0, r11, r4 @ r4 = nonyc_g = - cgu * (*U - 128) + r4 = - cgu * (*U - 128) - cgv * (*V - 128) + 32768 + add r0, pc, #(rb_clip-.-8) @ r0 contains the pointer to the R and B clipping array + mla r12, r8, r9, r5 @ r12 = (*Y - 16) * multy + nonyc_b + ldrb r7, [r0, r7, asr #(16+3)] @ r7 = R composant + mla r1, r8, r9, r4 @ r1 = (*Y - 16) * multy + nonyc_g + ldrb r9, [r10, #-1] @ r9 = *(Y+1) + str r10, [sp, #28] @ save Y + 2 + ldrb r12, [r0, r12, asr #(16+3)] @ r12 = B composant (and the start of the RGB word) + add r11, pc, #(g_clip-.-8) @ r11 now contains the pointer to the G clipping array + ldrb r1, [r11, r1, asr #(16+2)] @ r1 contains the G part of the RGB triplet + sub r9, r9, #16 @ r9 = *(Y+1) - 16 + mla r10, r8, r9, r6 @ r10 is the Red part of the RGB triplet + add r12, r12, r7, lsl #11 @ r12 = .GB ... + mla r7, r8, r9, r5 @ r7 is the Blue part of the RGB triplet + add r12, r12, r1, lsl #5 @ r12 = RGB ... (ie the first pixel (half-word) is done) + mla r2, r8, r9, r4 @ r2 is the Green part of the RGB triplet + ldrb r10, [r0, r10, asr #(16+3)] @ r10 = R composant + ldrb r7, [r0, r7, asr #(16+3)] @ r7 = B composant + ldr r1, [sp, #32] @ r1 = Ynext + ldrb r2, [r11, r2, asr #(16+2)] @ r2 = G composant + ldrb r9, [r1] @ r9 = *Ynext + add r12, r12, r2, lsl #(5+16) @ r12 = RGB .G. + sub r9, r9, #16 @ r9 = *Ynext - 16 + mla r2, r8, r9, r4 @ r2 is the Green part of the RGB triplet + add r12, r12, r7, lsl #(0+16) @ r12 = RGB .GB + mla r7, r8, r9, r5 @ r7 is the Blue part of the RGB triplet + add r12, r12, r10, lsl #(11+16) @ r12 = RGB RGB + ldr r3, [sp, #4] @ r3 = RGB + mla r10, r8, r9, r6 @ r10 is the Red part of the RGB triplet + str r12, [r3] @ store the rgb pixel at *RGB + add r3, r3, #4 @ r3 = RGB++ (ie next double-pixel) + str r3, [sp, #4] @ store the RGB pointer + ldrb r9, [r1, #1] @ r9 = *(Ynext+1) + add r1, r1, #2 @ r1 = Ynext + 2 + sub r9, r9, #16 @ r9 = *(Ynext+1) - 16 + ldrb r12, [r0, r7, asr #(16+3)] @ r12 = ..B ... + ldrb r10, [r0, r10, asr #(16+3)] @ r10 = B composant + mla r7, r8, r9, r5 @ r7 is the Blue part of the RGB triplet + add r12, r12, r10, lsl #11 @ r12 = R.B ... + ldrb r2, [r11, r2, asr #(16+2)] @ r2 = G composant + mla r10, r8, r9, r6 @ r10 is the Red part of the RGB triplet + add r12, r12, r2, lsl #5 @ r12 = RGB ... + mla r2, r8, r9, r4 @ r2 is the Green part of the RGB triplet + ldrb r7, [r0, r7, asr #(16+3)] @ r7 = B composant + str r1, [sp, #32] @ store the increased Ynext pointer + add r12, r12, r7, lsl #(16+0) @ r12 = RGB ..B + ldrb r10, [r0, r10, asr #(16+3)] @ r10 = R composant + ldr r3, [sp, #0] @ r3 = RGBnext pointer + add r12, r12, r10, lsl #(16+11) @ r12 = RGB R.B + ldrb r2, [r11, r2, asr #(16+2)] @ r2 = G composant + add r3, r3, #4 @ r3 = next pixel on the RGBnext line + add r12, r12, r2, lsl #(16+5) @ r12 = RGB RGB + str r12, [r3, #-4] @ store the next pixel + str r3, [sp, #0] @ store the increased 'next line' pixel pointer + subs lr, lr, #2 @ decrement the line counter + bne yuv_loop @ and restart if not at the end of the line + + ldr r0, [sp, #8] @ r0 = saved width + ldr r1, [sp, #0] @ r1 = RGBnext pointer + mov lr, r0 @ lr = saved width (to restart the line counter) + str r1, [sp, #4] @ current RGBnext pointer is next iteration RGB pointer + add r1, r1, r0, lsl #1 @ r1 = update RGBnext to next line + str r1, [sp, #0] @ store updated RGBnext pointer + + ldr r3, [sp, #16] @ r3 = (2 * stride_Y) - width + ldr r4, [sp, #28] @ r4 = Y ptr + ldr r5, [sp, #32] @ r5 = Ynext ptr + add r4, r4, r3 @ r4 = Y ptr for the next two lines + add r5, r5, r3 @ r5 = Ynext ptr for the next two lines + str r4, [sp, #28] @ store updated Y pointer + str r5, [sp, #32] @ store update Ynext pointer + + ldr r1, [sp, #20] @ r1 = stride_U - (width / 2) + ldr r2, [sp, #36] @ r2 = U ptr + + ldr r6, [sp, #12] @ get height counter + + add r2, r2, r1 @ update U ptr + str r2, [sp, #36] @ store updated U ptr (and update 'V' at the same time :-) ) + + subs r6, r6, #2 + str r6, [sp, #12] + bne yuv_loop + + @ Exit cleanly :-) + add sp, sp, #(11*4) @ remove all custom things from stack + ldmia sp!, { r4 - r12, pc } @ restore callee saved regs and return + + +const_storage: + @ In order : crv, cbu, - cgu, - cgv, multy + .word 0x00019895, 0x00020469, 0xffff9bb5, 0xffff2fe1, 0x00012A15 +rb_clip_dummy: + .byte 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00 + .byte 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00 + .byte 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00 +rb_clip: + .byte 0x00,0x01,0x02,0x03,0x04,0x05,0x06,0x07,0x08,0x09,0x0a,0x0b,0x0c,0x0d,0x0e,0x0f + .byte 0x10,0x11,0x12,0x13,0x14,0x15,0x16,0x17,0x18,0x19,0x1a,0x1b,0x1c,0x1d,0x1e,0x1f + .byte 0x1f,0x1f,0x1f,0x1f,0x1f,0x1f,0x1f,0x1f,0x1f,0x1f,0x1f,0x1f,0x1f,0x1f,0x1f,0x1f + .byte 0x1f,0x1f,0x1f,0x1f,0x1f,0x1f,0x1f,0x1f,0x1f,0x1f,0x1f,0x1f,0x1f,0x1f,0x1f,0x1f + .byte 0x1f,0x1f,0x1f,0x1f,0x1f,0x1f,0x1f,0x1f,0x1f,0x1f,0x1f,0x1f,0x1f,0x1f,0x1f,0x1f +g_clip_dummy: + .byte 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00 + .byte 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00 + .byte 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00 + .byte 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00 + .byte 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00 + .byte 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00 +g_clip: + .byte 0x00,0x01,0x02,0x03,0x04,0x05,0x06,0x07,0x08,0x09,0x0a,0x0b,0x0c,0x0d,0x0e,0x0f + .byte 0x10,0x11,0x12,0x13,0x14,0x15,0x16,0x17,0x18,0x19,0x1a,0x1b,0x1c,0x1d,0x1e,0x1f + .byte 0x20,0x21,0x22,0x23,0x24,0x25,0x26,0x27,0x28,0x29,0x2a,0x2b,0x2c,0x2d,0x2e,0x2f + .byte 0x30,0x31,0x32,0x33,0x34,0x35,0x36,0x37,0x38,0x39,0x3a,0x3b,0x3c,0x3d,0x3e,0x3f + .byte 0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f + .byte 0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f + .byte 0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f + .byte 0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f + .byte 0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f + .byte 0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f,0x3f + +#endif diff --git a/noncore/multimedia/opieplayer2/yuv2rgb_mlib.c b/noncore/multimedia/opieplayer2/yuv2rgb_mlib.c deleted file mode 100644 index 908b439..0000000 --- a/noncore/multimedia/opieplayer2/yuv2rgb_mlib.c +++ b/dev/null @@ -1,313 +0,0 @@ -/* - * yuv2rgb_mlib.c - * Copyright (C) 2000-2001 Silicon Integrated System Corp. - * All Rights Reserved. - * - * Author: Juergen Keil <jk@tools.de> - * - * This file is part of xine, a free unix video player. - * - * xine is free software; you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation; either version 2 of the License, or - * (at your option) any later version. - * - * xine is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program; if not, write to the Free Software - * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA - */ - - -#if HAVE_MLIB - -#include <stdio.h> -#include <stdlib.h> -#include <string.h> -#include <inttypes.h> -#include <mlib_video.h> - -#include "attributes.h" -#include "yuv2rgb.h" - - -static void scale_line (uint8_t *source, uint8_t *dest, - int width, int step) { - - unsigned p1; - unsigned p2; - int dx; - - p1 = *source++; - p2 = *source++; - dx = 0; - - while (width) { - - /* - printf ("scale_line, width = %d\n", width); - printf ("scale_line, dx = %d, p1 = %d, p2 = %d\n", dx, p1, p2); - */ - - *dest = (p1 * (32768 - dx) + p2 * dx) / 32768; - - dx += step; - while (dx > 32768) { - dx -= 32768; - p1 = p2; - p2 = *source++; - } - - dest ++; - width --; - } -} - - - -static void mlib_yuv420_rgb24 (yuv2rgb_t *this, - uint8_t * image, uint8_t * py, - uint8_t * pu, uint8_t * pv) -{ - int dst_height; - int dy; - mlib_status mlib_stat; - - if (this->do_scale) { - dy = 0; - dst_height = this->dest_height; - - for (;;) { - scale_line (pu, this->u_buffer, - this->dest_width >> 1, this->step_dx); - pu += this->uv_stride; - - scale_line (pv, this->v_buffer, - this->dest_width >> 1, this->step_dx); - pv += this->uv_stride; - - scale_line (py, this->y_buffer, - this->dest_width, this->step_dx); - py += this->y_stride; - scale_line (py, this->y_buffer + this->dest_width, - this->dest_width, this->step_dx); - py += this->y_stride; - - mlib_stat = mlib_VideoColorYUV2RGB420(image, - this->y_buffer, - this->u_buffer, - this->v_buffer, - this->dest_width & ~1, 2, - this->rgb_stride, - this->dest_width, - this->dest_width >> 1); - dy += this->step_dy; - image += this->rgb_stride; - - while (--dst_height > 0 && dy < 32768) { - memcpy (image, (uint8_t*)image-this->rgb_stride, this->dest_width*6); - dy += this->step_dy; - image += this->rgb_stride; - } - - if (dst_height <= 0) - break; - - dy -= 32768; - - dy += this->step_dy; - image += this->rgb_stride; - - while (--dst_height > 0 && dy < 32768) { - memcpy (image, (uint8_t*)image-this->rgb_stride, this->dest_width*3); - dy += this->step_dy; - image += this->rgb_stride; - } - - if (dst_height <= 0) - break; - - dy -= 32768; - } - } else { - mlib_stat = mlib_VideoColorYUV2RGB420(image, py, pu, pv, - this->source_width, - this->source_height, - this->rgb_stride, - this->y_stride, - this->uv_stride); - } -} - -static void mlib_yuv420_argb32 (yuv2rgb_t *this, - uint8_t * image, uint8_t * py, - uint8_t * pu, uint8_t * pv) -{ - int dst_height; - int dy; - mlib_status mlib_stat; - - if (this->do_scale) { - dy = 0; - dst_height = this->dest_height; - - for (;;) { - scale_line (pu, this->u_buffer, - this->dest_width >> 1, this->step_dx); - pu += this->uv_stride; - - scale_line (pv, this->v_buffer, - this->dest_width >> 1, this->step_dx); - pv += this->uv_stride; - - scale_line (py, this->y_buffer, - this->dest_width, this->step_dx); - py += this->y_stride; - scale_line (py, this->y_buffer + this->dest_width, - this->dest_width, this->step_dx); - py += this->y_stride; - - mlib_stat = mlib_VideoColorYUV2ARGB420(image, - this->y_buffer, - this->u_buffer, - this->v_buffer, - this->dest_width & ~1, 2, - this->rgb_stride, - this->dest_width, - this->dest_width >> 1); - dy += this->step_dy; - image += this->rgb_stride; - - while (--dst_height > 0 && dy < 32768) { - memcpy (image, (uint8_t*)image-this->rgb_stride, this->dest_width*8); - dy += this->step_dy; - image += this->rgb_stride; - } - - if (dst_height <= 0) - break; - - dy -= 32768; - - dy += this->step_dy; - image += this->rgb_stride; - - while (--dst_height > 0 && dy < 32768) { - memcpy (image, (uint8_t*)image-this->rgb_stride, this->dest_width*4); - dy += this->step_dy; - image += this->rgb_stride; - } - - if (dst_height <= 0) - break; - - dy -= 32768; - } - } else { - mlib_stat = mlib_VideoColorYUV2ARGB420(image, py, pu, pv, - this->source_width, - this->source_height, - this->rgb_stride, - this->y_stride, - this->uv_stride); - } -} - -static void mlib_yuv420_abgr32 (yuv2rgb_t *this, - uint8_t * image, uint8_t * py, - uint8_t * pu, uint8_t * pv) -{ - int dst_height; - int dy; - mlib_status mlib_stat; - - if (this->do_scale) { - dy = 0; - dst_height = this->dest_height; - - for (;;) { - scale_line (pu, this->u_buffer, - this->dest_width >> 1, this->step_dx); - pu += this->uv_stride; - - scale_line (pv, this->v_buffer, - this->dest_width >> 1, this->step_dx); - pv += this->uv_stride; - - scale_line (py, this->y_buffer, - this->dest_width, this->step_dx); - py += this->y_stride; - scale_line (py, this->y_buffer + this->dest_width, - this->dest_width, this->step_dx); - py += this->y_stride; - - mlib_stat = mlib_VideoColorYUV2ABGR420(image, - this->y_buffer, - this->u_buffer, - this->v_buffer, - this->dest_width & ~1, 2, - this->rgb_stride, - this->dest_width, - this->dest_width >> 1); - dy += this->step_dy; - image += this->rgb_stride; - - while (--dst_height > 0 && dy < 32768) { - memcpy (image, (uint8_t*)image-this->rgb_stride, this->dest_width*8); - dy += this->step_dy; - image += this->rgb_stride; - } - - if (dst_height <= 0) - break; - - dy -= 32768; - - dy += this->step_dy; - image += this->rgb_stride; - - while (--dst_height > 0 && dy < 32768) { - memcpy (image, (uint8_t*)image-this->rgb_stride, this->dest_width*4); - dy += this->step_dy; - image += this->rgb_stride; - } - - if (dst_height <= 0) - break; - - dy -= 32768; - } - } else { - mlib_stat = mlib_VideoColorYUV2ABGR420(image, py, pu, pv, - this->source_width, - this->source_height, - this->rgb_stride, - this->y_stride, - this->uv_stride); - } -} - - -void yuv2rgb_init_mlib (yuv2rgb_factory_t *this) { - - if (this->swapped) return; /*no swapped pixel output upto now*/ - - switch (this->mode) { - case MODE_24_RGB: - this->yuv2rgb_fun = mlib_yuv420_rgb24; - break; - case MODE_32_RGB: - this->yuv2rgb_fun = mlib_yuv420_argb32; - break; - case MODE_32_BGR: - this->yuv2rgb_fun = mlib_yuv420_abgr32; - break; - } -} - - -#endif /* HAVE_MLIB */ diff --git a/noncore/multimedia/opieplayer2/yuv2rgb_mmx.c b/noncore/multimedia/opieplayer2/yuv2rgb_mmx.c deleted file mode 100644 index f092e6f..0000000 --- a/noncore/multimedia/opieplayer2/yuv2rgb_mmx.c +++ b/dev/null @@ -1,1047 +0,0 @@ -/* - * yuv2rgb_mmx.c - * Copyright (C) 2000-2001 Silicon Integrated System Corp. - * All Rights Reserved. - * - * Author: Olie Lho <ollie@sis.com.tw> - * - * This file is part of mpeg2dec, a free MPEG-2 video stream decoder. - * - * mpeg2dec is free software; you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation; either version 2 of the License, or - * (at your option) any later version. - * - * mpeg2dec is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program; if not, write to the Free Software - * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA - */ - - -#ifdef ARCH_X86 - -#include <stdio.h> -#include <stdlib.h> -#include <string.h> -#include <inttypes.h> - -#include "yuv2rgb.h" -#include "xineutils.h" - -#define CPU_MMXEXT 0 -#define CPU_MMX 1 - -/* CPU_MMXEXT/CPU_MMX adaptation layer */ - -#define movntq(src,dest) \ -do { \ - if (cpu == CPU_MMXEXT) \ - movntq_r2m (src, dest); \ - else \ - movq_r2m (src, dest); \ -} while (0) - -static mmx_t mmx_subYw = {0x1010101010101010}; -static mmx_t mmx_addYw = {0x0000000000000000}; - -void mmx_yuv2rgb_set_gamma(int gamma) -{ -int a,s,i; - - if( gamma <= 16 ) { - a = 0; - s = 16 - gamma; - } else { - a = gamma - 16; - s = 0; - } - - for( i = 0; i < 8; i++ ) { - *((unsigned char *)&mmx_subYw + i) = s; - *((unsigned char *)&mmx_addYw + i) = a; - } -} - -static inline void mmx_yuv2rgb (uint8_t * py, uint8_t * pu, uint8_t * pv) -{ - static mmx_t mmx_80w = {0x0080008000800080}; - static mmx_t mmx_U_green = {0xf37df37df37df37d}; - static mmx_t mmx_U_blue = {0x4093409340934093}; - static mmx_t mmx_V_red = {0x3312331233123312}; - static mmx_t mmx_V_green = {0xe5fce5fce5fce5fc}; - static mmx_t mmx_00ffw = {0x00ff00ff00ff00ff}; - static mmx_t mmx_Y_coeff = {0x253f253f253f253f}; - - movq_m2r (*py, mm6); // mm6 = Y7 Y6 Y5 Y4 Y3 Y2 Y1 Y0 - pxor_r2r (mm4, mm4); // mm4 = 0 - - psubusb_m2r (mmx_subYw, mm6); // Y -= 16 - paddusb_m2r (mmx_addYw, mm6); - - movd_m2r (*pu, mm0); // mm0 = 00 00 00 00 u3 u2 u1 u0 - movq_r2r (mm6, mm7); // mm7 = Y7 Y6 Y5 Y4 Y3 Y2 Y1 Y0 - - pand_m2r (mmx_00ffw, mm6); // mm6 = Y6 Y4 Y2 Y0 - psrlw_i2r (8, mm7); // mm7 = Y7 Y5 Y3 Y1 - - movd_m2r (*pv, mm1); // mm1 = 00 00 00 00 v3 v2 v1 v0 - psllw_i2r (3, mm6); // promote precision - - pmulhw_m2r (mmx_Y_coeff, mm6); // mm6 = luma_rgb even - psllw_i2r (3, mm7); // promote precision - - punpcklbw_r2r (mm4, mm0); // mm0 = u3 u2 u1 u0 - - psubsw_m2r (mmx_80w, mm0); // u -= 128 - punpcklbw_r2r (mm4, mm1); // mm1 = v3 v2 v1 v0 - - pmulhw_m2r (mmx_Y_coeff, mm7); // mm7 = luma_rgb odd - psllw_i2r (3, mm0); // promote precision - - psubsw_m2r (mmx_80w, mm1); // v -= 128 - movq_r2r (mm0, mm2); // mm2 = u3 u2 u1 u0 - - psllw_i2r (3, mm1); // promote precision - - movq_r2r (mm1, mm4); // mm4 = v3 v2 v1 v0 - - pmulhw_m2r (mmx_U_blue, mm0); // mm0 = chroma_b - - - // slot - - - // slot - - - pmulhw_m2r (mmx_V_red, mm1); // mm1 = chroma_r - movq_r2r (mm0, mm3); // mm3 = chroma_b - - paddsw_r2r (mm6, mm0); // mm0 = B6 B4 B2 B0 - paddsw_r2r (mm7, mm3); // mm3 = B7 B5 B3 B1 - - packuswb_r2r (mm0, mm0); // saturate to 0-255 - - - pmulhw_m2r (mmx_U_green, mm2); // mm2 = u * u_green - - - packuswb_r2r (mm3, mm3); // saturate to 0-255 - - - punpcklbw_r2r (mm3, mm0); // mm0 = B7 B6 B5 B4 B3 B2 B1 B0 - - - pmulhw_m2r (mmx_V_green, mm4); // mm4 = v * v_green - - - // slot - - - // slot - - - paddsw_r2r (mm4, mm2); // mm2 = chroma_g - movq_r2r (mm2, mm5); // mm5 = chroma_g - - - movq_r2r (mm1, mm4); // mm4 = chroma_r - paddsw_r2r (mm6, mm2); // mm2 = G6 G4 G2 G0 - - - packuswb_r2r (mm2, mm2); // saturate to 0-255 - paddsw_r2r (mm6, mm1); // mm1 = R6 R4 R2 R0 - - packuswb_r2r (mm1, mm1); // saturate to 0-255 - paddsw_r2r (mm7, mm4); // mm4 = R7 R5 R3 R1 - - packuswb_r2r (mm4, mm4); // saturate to 0-255 - paddsw_r2r (mm7, mm5); // mm5 = G7 G5 G3 G1 - - - packuswb_r2r (mm5, mm5); // saturate to 0-255 - - - punpcklbw_r2r (mm4, mm1); // mm1 = R7 R6 R5 R4 R3 R2 R1 R0 - - - punpcklbw_r2r (mm5, mm2); // mm2 = G7 G6 G5 G4 G3 G2 G1 G0 -} - -// basic opt -static inline void mmx_unpack_16rgb (uint8_t * image, int cpu) -{ - static mmx_t mmx_bluemask = {0xf8f8f8f8f8f8f8f8}; - static mmx_t mmx_greenmask = {0xfcfcfcfcfcfcfcfc}; - static mmx_t mmx_redmask = {0xf8f8f8f8f8f8f8f8}; - - /* - * convert RGB plane to RGB 16 bits - * mm0 -> B, mm1 -> R, mm2 -> G - * mm4 -> GB, mm5 -> AR pixel 4-7 - * mm6 -> GB, mm7 -> AR pixel 0-3 - */ - - pand_m2r (mmx_bluemask, mm0); // mm0 = b7b6b5b4b3______ - pxor_r2r (mm4, mm4); // mm4 = 0 - - pand_m2r (mmx_greenmask, mm2); // mm2 = g7g6g5g4g3g2____ - psrlq_i2r (3, mm0); // mm0 = ______b7b6b5b4b3 - - movq_r2r (mm2, mm7); // mm7 = g7g6g5g4g3g2____ - movq_r2r (mm0, mm5); // mm5 = ______b7b6b5b4b3 - - pand_m2r (mmx_redmask, mm1); // mm1 = r7r6r5r4r3______ - punpcklbw_r2r (mm4, mm2); - - punpcklbw_r2r (mm1, mm0); - - psllq_i2r (3, mm2); - - punpckhbw_r2r (mm4, mm7); - por_r2r (mm2, mm0); - - psllq_i2r (3, mm7); - - movntq (mm0, *image); - punpckhbw_r2r (mm1, mm5); - - por_r2r (mm7, mm5); - - // U - // V - - movntq (mm5, *(image+8)); -} - -static inline void mmx_unpack_15rgb (uint8_t * image, int cpu) -{ - static mmx_t mmx_bluemask = {0xf8f8f8f8f8f8f8f8}; - static mmx_t mmx_greenmask = {0xf8f8f8f8f8f8f8f8}; - static mmx_t mmx_redmask = {0xf8f8f8f8f8f8f8f8}; - - /* - * convert RGB plane to RGB 15 bits - * mm0 -> B, mm1 -> R, mm2 -> G - * mm4 -> GB, mm5 -> AR pixel 4-7 - * mm6 -> GB, mm7 -> AR pixel 0-3 - */ - - pand_m2r (mmx_bluemask, mm0); // mm0 = b7b6b5b4b3______ - pxor_r2r (mm4, mm4); // mm4 = 0 - - pand_m2r (mmx_greenmask, mm2); // mm2 = g7g6g5g4g3g2____ - psrlq_i2r (3, mm0); // mm0 = ______b7b6b5b4b3 - - movq_r2r (mm2, mm7); // mm7 = g7g6g5g4g3g2____ - movq_r2r (mm0, mm5); // mm5 = ______b7b6b5b4b3 - - pand_m2r (mmx_redmask, mm1); // mm1 = r7r6r5r4r3______ - punpcklbw_r2r (mm4, mm2); - - psrlq_i2r (1, mm1); - punpcklbw_r2r (mm1, mm0); - - psllq_i2r (2, mm2); - - punpckhbw_r2r (mm4, mm7); - por_r2r (mm2, mm0); - - psllq_i2r (2, mm7); - - movntq (mm0, *image); - punpckhbw_r2r (mm1, mm5); - - por_r2r (mm7, mm5); - - // U - // V - - movntq (mm5, *(image+8)); -} - -static inline void mmx_unpack_32rgb (uint8_t * image, int cpu) -{ - /* - * convert RGB plane to RGB packed format, - * mm0 -> B, mm1 -> R, mm2 -> G, mm3 -> 0, - * mm4 -> GB, mm5 -> AR pixel 4-7, - * mm6 -> GB, mm7 -> AR pixel 0-3 - */ - - pxor_r2r (mm3, mm3); - movq_r2r (mm0, mm6); - - punpcklbw_r2r (mm2, mm6); - movq_r2r (mm1, mm7); - - punpcklbw_r2r (mm3, mm7); - movq_r2r (mm0, mm4); - - punpcklwd_r2r (mm7, mm6); - movq_r2r (mm1, mm5); - - /* scheduling: this is hopeless */ - movntq (mm6, *image); - movq_r2r (mm0, mm6); - punpcklbw_r2r (mm2, mm6); - punpckhwd_r2r (mm7, mm6); - movntq (mm6, *(image+8)); - punpckhbw_r2r (mm2, mm4); - punpckhbw_r2r (mm3, mm5); - punpcklwd_r2r (mm5, mm4); - movntq (mm4, *(image+16)); - movq_r2r (mm0, mm4); - punpckhbw_r2r (mm2, mm4); - punpckhwd_r2r (mm5, mm4); - movntq (mm4, *(image+24)); -} - -static inline void mmx_unpack_32bgr (uint8_t * image, int cpu) -{ - /* - * convert RGB plane to RGB packed format, - * mm0 -> B, mm1 -> R, mm2 -> G, mm3 -> 0, - * mm4 -> GB, mm5 -> AR pixel 4-7, - * mm6 -> GB, mm7 -> AR pixel 0-3 - */ - - pxor_r2r (mm3, mm3); - movq_r2r (mm1, mm6); - - punpcklbw_r2r (mm2, mm6); - movq_r2r (mm0, mm7); - - punpcklbw_r2r (mm3, mm7); - movq_r2r (mm1, mm4); - - punpcklwd_r2r (mm7, mm6); - movq_r2r (mm0, mm5); - - /* scheduling: this is hopeless */ - movntq (mm6, *image); - movq_r2r (mm0, mm6); - punpcklbw_r2r (mm2, mm6); - punpckhwd_r2r (mm7, mm6); - movntq (mm6, *(image+8)); - punpckhbw_r2r (mm2, mm4); - punpckhbw_r2r (mm3, mm5); - punpcklwd_r2r (mm5, mm4); - movntq (mm4, *(image+16)); - movq_r2r (mm0, mm4); - punpckhbw_r2r (mm2, mm4); - punpckhwd_r2r (mm5, mm4); - movntq (mm4, *(image+24)); -} - -static inline void mmx_unpack_24rgb (uint8_t * image, int cpu) -{ - /* - * convert RGB plane to RGB packed format, - * mm0 -> B, mm1 -> R, mm2 -> G, mm3 -> 0, - * mm4 -> GB, mm5 -> AR pixel 4-7, - * mm6 -> GB, mm7 -> AR pixel 0-3 - */ - - pxor_r2r (mm3, mm3); - movq_r2r (mm0, mm6); - - punpcklbw_r2r (mm2, mm6); - movq_r2r (mm1, mm7); - - punpcklbw_r2r (mm3, mm7); - movq_r2r (mm0, mm4); - - punpcklwd_r2r (mm7, mm6); - movq_r2r (mm1, mm5); - - /* scheduling: this is hopeless */ - movntq (mm6, *image); - movq_r2r (mm0, mm6); - punpcklbw_r2r (mm2, mm6); - punpckhwd_r2r (mm7, mm6); - movntq (mm6, *(image+8)); - punpckhbw_r2r (mm2, mm4); - punpckhbw_r2r (mm3, mm5); - punpcklwd_r2r (mm5, mm4); - movntq (mm4, *(image+16)); -} - -static inline void yuv420_rgb16 (yuv2rgb_t *this, - uint8_t * image, - uint8_t * py, uint8_t * pu, uint8_t * pv, - int cpu) -{ - int i; - int rgb_stride = this->rgb_stride; - int y_stride = this->y_stride; - int uv_stride = this->uv_stride; - int width = this->source_width; - int height = this->source_height; - int dst_height = this->dest_height; - uint8_t *img; - - width >>= 3; - - if (!this->do_scale) { - y_stride -= 8 * width; - uv_stride -= 4 * width; - - do { - - i = width; img = image; - do { - mmx_yuv2rgb (py, pu, pv); - mmx_unpack_16rgb (img, cpu); - py += 8; - pu += 4; - pv += 4; - img += 16; - } while (--i); - - py += y_stride; - image += rgb_stride; - if (height & 1) { - pu += uv_stride; - pv += uv_stride; - } else { - pu -= 4 * width; - pv -= 4 * width; - } - } while (--height); - - } else { - - scale_line_func_t scale_line = this->scale_line; - uint8_t *y_buf, *u_buf, *v_buf; - int dy = 0; - - scale_line (pu, this->u_buffer, - this->dest_width >> 1, this->step_dx); - scale_line (pv, this->v_buffer, - this->dest_width >> 1, this->step_dx); - scale_line (py, this->y_buffer, - this->dest_width, this->step_dx); - for (height = 0;; ) { - - y_buf = this->y_buffer; - u_buf = this->u_buffer; - v_buf = this->v_buffer; - - i = this->dest_width >> 3; img = image; - do { - /* printf ("i : %d\n",i); */ - - mmx_yuv2rgb (y_buf, u_buf, v_buf); - mmx_unpack_16rgb (img, cpu); - y_buf += 8; - u_buf += 4; - v_buf += 4; - img += 16; - } while (--i); - - dy += this->step_dy; - image += rgb_stride; - - while (--dst_height > 0 && dy < 32768) { - - xine_fast_memcpy (image, image-rgb_stride, this->dest_width*2); - - dy += this->step_dy; - image += rgb_stride; - } - - if (dst_height <= 0) - break; - - do { - dy -= 32768; - - py += y_stride; - - scale_line (py, this->y_buffer, - this->dest_width, this->step_dx); - - if (height & 1) { - pu += uv_stride; - pv += uv_stride; - - scale_line (pu, this->u_buffer, - this->dest_width >> 1, this->step_dx); - scale_line (pv, this->v_buffer, - this->dest_width >> 1, this->step_dx); - - } - height++; - } while( dy>=32768); - } - } -} - -static inline void yuv420_rgb15 (yuv2rgb_t *this, - uint8_t * image, - uint8_t * py, uint8_t * pu, uint8_t * pv, - int cpu) -{ - int i; - int rgb_stride = this->rgb_stride; - int y_stride = this->y_stride; - int uv_stride = this->uv_stride; - int width = this->source_width; - int height = this->source_height; - int dst_height = this->dest_height; - uint8_t *img; - - width >>= 3; - - if (!this->do_scale) { - y_stride -= 8 * width; - uv_stride -= 4 * width; - - do { - - i = width; img = image; - do { - mmx_yuv2rgb (py, pu, pv); - mmx_unpack_15rgb (img, cpu); - py += 8; - pu += 4; - pv += 4; - img += 16; - } while (--i); - - py += y_stride; - image += rgb_stride; - if (height & 1) { - pu += uv_stride; - pv += uv_stride; - } else { - pu -= 4 * width; - pv -= 4 * width; - } - } while (--height); - - } else { - - scale_line_func_t scale_line = this->scale_line; - uint8_t *y_buf, *u_buf, *v_buf; - int dy = 0; - - scale_line (pu, this->u_buffer, - this->dest_width >> 1, this->step_dx); - scale_line (pv, this->v_buffer, - this->dest_width >> 1, this->step_dx); - scale_line (py, this->y_buffer, - this->dest_width, this->step_dx); - for (height = 0;; ) { - - y_buf = this->y_buffer; - u_buf = this->u_buffer; - v_buf = this->v_buffer; - - i = this->dest_width >> 3; img = image; - do { - /* printf ("i : %d\n",i); */ - - mmx_yuv2rgb (y_buf, u_buf, v_buf); - mmx_unpack_15rgb (img, cpu); - y_buf += 8; - u_buf += 4; - v_buf += 4; - img += 16; - } while (--i); - - dy += this->step_dy; - image += rgb_stride; - - while (--dst_height > 0 && dy < 32768) { - - xine_fast_memcpy (image, image-rgb_stride, this->dest_width*2); - - dy += this->step_dy; - image += rgb_stride; - } - - if (dst_height <= 0) - break; - - do { - dy -= 32768; - py += y_stride; - - scale_line (py, this->y_buffer, - this->dest_width, this->step_dx); - - if (height & 1) { - pu += uv_stride; - pv += uv_stride; - - scale_line (pu, this->u_buffer, - this->dest_width >> 1, this->step_dx); - scale_line (pv, this->v_buffer, - this->dest_width >> 1, this->step_dx); - - } - height++; - } while( dy>=32768 ); - } - } -} - -static inline void yuv420_rgb24 (yuv2rgb_t *this, - uint8_t * image, uint8_t * py, - uint8_t * pu, uint8_t * pv, int cpu) -{ - int i; - int rgb_stride = this->rgb_stride; - int y_stride = this->y_stride; - int uv_stride = this->uv_stride; - int width = this->source_width; - int height = this->source_height; - int dst_height = this->dest_height; - uint8_t *img; - - /* rgb_stride -= 4 * this->dest_width; */ - width >>= 3; - - if (!this->do_scale) { - y_stride -= 8 * width; - uv_stride -= 4 * width; - - do { - i = width; img = image; - do { - mmx_yuv2rgb (py, pu, pv); - mmx_unpack_24rgb (img, cpu); - py += 8; - pu += 4; - pv += 4; - img += 24; - } while (--i); - - py += y_stride; - image += rgb_stride; - if (height & 1) { - pu += uv_stride; - pv += uv_stride; - } else { - pu -= 4 * width; - pv -= 4 * width; - } - } while (--height); - } else { - - scale_line_func_t scale_line = this->scale_line; - uint8_t *y_buf, *u_buf, *v_buf; - int dy = 0; - - scale_line (pu, this->u_buffer, - this->dest_width >> 1, this->step_dx); - scale_line (pv, this->v_buffer, - this->dest_width >> 1, this->step_dx); - scale_line (py, this->y_buffer, - this->dest_width, this->step_dx); - - for (height = 0;; ) { - - y_buf = this->y_buffer; - u_buf = this->u_buffer; - v_buf = this->v_buffer; - - - i = this->dest_width >> 3; img=image; - do { - /* printf ("i : %d\n",i); */ - - mmx_yuv2rgb (y_buf, u_buf, v_buf); - mmx_unpack_24rgb (img, cpu); - y_buf += 8; - u_buf += 4; - v_buf += 4; - img += 24; - } while (--i); - - dy += this->step_dy; - image += rgb_stride; - - while (--dst_height > 0 && dy < 32768) { - - xine_fast_memcpy (image, image-rgb_stride, this->dest_width*3); - - dy += this->step_dy; - image += rgb_stride; - } - - if (dst_height <= 0) - break; - - do { - dy -= 32768; - py += y_stride; - - scale_line (py, this->y_buffer, - this->dest_width, this->step_dx); - - if (height & 1) { - pu += uv_stride; - pv += uv_stride; - - scale_line (pu, this->u_buffer, - this->dest_width >> 1, this->step_dx); - scale_line (pv, this->v_buffer, - this->dest_width >> 1, this->step_dx); - } - height++; - } while( dy>=32768 ); - - } - - } -} - -static inline void yuv420_argb32 (yuv2rgb_t *this, - uint8_t * image, uint8_t * py, - uint8_t * pu, uint8_t * pv, int cpu) -{ - int i; - int rgb_stride = this->rgb_stride; - int y_stride = this->y_stride; - int uv_stride = this->uv_stride; - int width = this->source_width; - int height = this->source_height; - int dst_height = this->dest_height; - uint8_t *img; - - /* rgb_stride -= 4 * this->dest_width; */ - width >>= 3; - - if (!this->do_scale) { - y_stride -= 8 * width; - uv_stride -= 4 * width; - - do { - i = width; img = image; - do { - mmx_yuv2rgb (py, pu, pv); - mmx_unpack_32rgb (img, cpu); - py += 8; - pu += 4; - pv += 4; - img += 32; - } while (--i); - - py += y_stride; - image += rgb_stride; - if (height & 1) { - pu += uv_stride; - pv += uv_stride; - } else { - pu -= 4 * width; - pv -= 4 * width; - } - } while (--height); - } else { - - scale_line_func_t scale_line = this->scale_line; - uint8_t *y_buf, *u_buf, *v_buf; - int dy = 0; - - scale_line (pu, this->u_buffer, - this->dest_width >> 1, this->step_dx); - scale_line (pv, this->v_buffer, - this->dest_width >> 1, this->step_dx); - scale_line (py, this->y_buffer, - this->dest_width, this->step_dx); - - for (height = 0;; ) { - - y_buf = this->y_buffer; - u_buf = this->u_buffer; - v_buf = this->v_buffer; - - - i = this->dest_width >> 3; img=image; - do { - /* printf ("i : %d\n",i); */ - - mmx_yuv2rgb (y_buf, u_buf, v_buf); - mmx_unpack_32rgb (img, cpu); - y_buf += 8; - u_buf += 4; - v_buf += 4; - img += 32; - } while (--i); - - dy += this->step_dy; - image += rgb_stride; - - while (--dst_height > 0 && dy < 32768) { - - xine_fast_memcpy (image, image-rgb_stride, this->dest_width*4); - - dy += this->step_dy; - image += rgb_stride; - } - - if (dst_height <= 0) - break; - - do { - dy -= 32768; - py += y_stride; - - scale_line (py, this->y_buffer, - this->dest_width, this->step_dx); - - if (height & 1) { - pu += uv_stride; - pv += uv_stride; - - scale_line (pu, this->u_buffer, - this->dest_width >> 1, this->step_dx); - scale_line (pv, this->v_buffer, - this->dest_width >> 1, this->step_dx); - } - height++; - } while( dy>=32768 ); - } - - } -} - -static inline void yuv420_abgr32 (yuv2rgb_t *this, - uint8_t * image, uint8_t * py, - uint8_t * pu, uint8_t * pv, int cpu) -{ - int i; - int rgb_stride = this->rgb_stride; - int y_stride = this->y_stride; - int uv_stride = this->uv_stride; - int width = this->source_width; - int height = this->source_height; - int dst_height = this->dest_height; - uint8_t *img; - - /* rgb_stride -= 4 * this->dest_width; */ - width >>= 3; - - if (!this->do_scale) { - y_stride -= 8 * width; - uv_stride -= 4 * width; - - do { - i = width; img = image; - do { - mmx_yuv2rgb (py, pu, pv); - mmx_unpack_32bgr (img, cpu); - py += 8; - pu += 4; - pv += 4; - img += 32; - } while (--i); - - py += y_stride; - image += rgb_stride; - if (height & 1) { - pu += uv_stride; - pv += uv_stride; - } else { - pu -= 4 * width; - pv -= 4 * width; - } - } while (--height); - } else { - - scale_line_func_t scale_line = this->scale_line; - uint8_t *y_buf, *u_buf, *v_buf; - int dy = 0; - - scale_line (pu, this->u_buffer, - this->dest_width >> 1, this->step_dx); - scale_line (pv, this->v_buffer, - this->dest_width >> 1, this->step_dx); - scale_line (py, this->y_buffer, - this->dest_width, this->step_dx); - - for (height = 0;; ) { - - y_buf = this->y_buffer; - u_buf = this->u_buffer; - v_buf = this->v_buffer; - - - i = this->dest_width >> 3; img=image; - do { - /* printf ("i : %d\n",i); */ - - mmx_yuv2rgb (y_buf, u_buf, v_buf); - mmx_unpack_32bgr (img, cpu); - y_buf += 8; - u_buf += 4; - v_buf += 4; - img += 32; - } while (--i); - - dy += this->step_dy; - image += rgb_stride; - - while (--dst_height > 0 && dy < 32768) { - - xine_fast_memcpy (image, image-rgb_stride, this->dest_width*4); - - dy += this->step_dy; - image += rgb_stride; - } - - if (dst_height <= 0) - break; - - do { - dy -= 32768; - py += y_stride; - - scale_line (py, this->y_buffer, - this->dest_width, this->step_dx); - - if (height & 1) { - pu += uv_stride; - pv += uv_stride; - - scale_line (pu, this->u_buffer, - this->dest_width >> 1, this->step_dx); - scale_line (pv, this->v_buffer, - this->dest_width >> 1, this->step_dx); - } - height++; - } while( dy>=32768 ); - - } - - } -} - -static void mmxext_rgb15 (yuv2rgb_t *this, uint8_t * image, - uint8_t * py, uint8_t * pu, uint8_t * pv) -{ - yuv420_rgb15 (this, image, py, pu, pv, CPU_MMXEXT); - emms(); /* re-initialize x86 FPU after MMX use */ -} - -static void mmxext_rgb16 (yuv2rgb_t *this, uint8_t * image, - uint8_t * py, uint8_t * pu, uint8_t * pv) -{ - yuv420_rgb16 (this, image, py, pu, pv, CPU_MMXEXT); - emms(); /* re-initialize x86 FPU after MMX use */ -} - -static void mmxext_rgb24 (yuv2rgb_t *this, uint8_t * image, - uint8_t * py, uint8_t * pu, uint8_t * pv) -{ - yuv420_rgb24 (this, image, py, pu, pv, CPU_MMXEXT); - emms(); /* re-initialize x86 FPU after MMX use */ -} - -static void mmxext_argb32 (yuv2rgb_t *this, uint8_t * image, - uint8_t * py, uint8_t * pu, uint8_t * pv) -{ - yuv420_argb32 (this, image, py, pu, pv, CPU_MMXEXT); - emms(); /* re-initialize x86 FPU after MMX use */ -} - -static void mmxext_abgr32 (yuv2rgb_t *this, uint8_t * image, - uint8_t * py, uint8_t * pu, uint8_t * pv) -{ - yuv420_abgr32 (this, image, py, pu, pv, CPU_MMXEXT); - emms(); /* re-initialize x86 FPU after MMX use */ -} - -static void mmx_rgb15 (yuv2rgb_t *this, uint8_t * image, - uint8_t * py, uint8_t * pu, uint8_t * pv) -{ - yuv420_rgb15 (this, image, py, pu, pv, CPU_MMX); - emms(); /* re-initialize x86 FPU after MMX use */ -} - -static void mmx_rgb16 (yuv2rgb_t *this, uint8_t * image, - uint8_t * py, uint8_t * pu, uint8_t * pv) -{ - yuv420_rgb16 (this, image, py, pu, pv, CPU_MMX); - emms(); /* re-initialize x86 FPU after MMX use */ -} - -static void mmx_rgb24 (yuv2rgb_t *this, uint8_t * image, - uint8_t * py, uint8_t * pu, uint8_t * pv) -{ - yuv420_rgb24 (this, image, py, pu, pv, CPU_MMX); - emms(); /* re-initialize x86 FPU after MMX use */ -} - -static void mmx_argb32 (yuv2rgb_t *this, uint8_t * image, - uint8_t * py, uint8_t * pu, uint8_t * pv) -{ - yuv420_argb32 (this, image, py, pu, pv, CPU_MMX); - emms(); /* re-initialize x86 FPU after MMX use */ -} - -static void mmx_abgr32 (yuv2rgb_t *this, uint8_t * image, - uint8_t * py, uint8_t * pu, uint8_t * pv) -{ - yuv420_abgr32 (this, image, py, pu, pv, CPU_MMX); - emms(); /* re-initialize x86 FPU after MMX use */ -} - -void yuv2rgb_init_mmxext (yuv2rgb_factory_t *this) { - - if (this->swapped) - return; /*no swapped pixel output upto now*/ - - switch (this->mode) { - case MODE_15_RGB: - this->yuv2rgb_fun = mmxext_rgb15; - break; - case MODE_16_RGB: - this->yuv2rgb_fun = mmxext_rgb16; - break; - case MODE_24_RGB: - this->yuv2rgb_fun = mmxext_rgb24; - break; - case MODE_32_RGB: - this->yuv2rgb_fun = mmxext_argb32; - break; - case MODE_32_BGR: - this->yuv2rgb_fun = mmxext_abgr32; - break; - } -} - -void yuv2rgb_init_mmx (yuv2rgb_factory_t *this) { - - if (this->swapped) - return; /*no swapped pixel output upto now*/ - - switch (this->mode) { - case MODE_15_RGB: - this->yuv2rgb_fun = mmx_rgb15; - break; - case MODE_16_RGB: - this->yuv2rgb_fun = mmx_rgb16; - break; - case MODE_24_RGB: - this->yuv2rgb_fun = mmx_rgb24; - break; - case MODE_32_RGB: - this->yuv2rgb_fun = mmx_argb32; - break; - case MODE_32_BGR: - this->yuv2rgb_fun = mmx_abgr32; - break; - } -} - - -#endif |