/* * Copyright (C) 1994-2000 The XFree86 Project, Inc. All Rights Reserved. * Copyright (c) 2003-2006, X.Org Foundation * * Permission is hereby granted, free of charge, to any person obtaining a * copy of this software and associated documentation files (the "Software"), * to deal in the Software without restriction, including without limitation * the rights to use, copy, modify, merge, publish, distribute, sublicense, * and/or sell copies of the Software, and to permit persons to whom the * Software is furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included in * all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE * COPYRIGHT HOLDER(S) OR AUTHOR(S) BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER * DEALINGS IN THE SOFTWARE. * * Except as contained in this notice, the name of the copyright holder(s) * and author(s) shall not be used in advertising or otherwise to promote * the sale, use or other dealings in this Software without prior written * authorization from the copyright holder(s) and author(s). */ #ifdef HAVE_CONFIG_H #include "config.h" #endif #include #include "dix.h" #include "dixstruct.h" #include "fourcc.h" #include "savage_driver.h" #include "savage_streams.h" #include "savage_regs.h" #include "savage_bci.h" #define OFF_DELAY 200 /* milliseconds */ #define FREE_DELAY 60000 #define OFF_TIMER 0x01 #define FREE_TIMER 0x02 #define CLIENT_VIDEO_ON 0x04 #define TIMER_MASK (OFF_TIMER | FREE_TIMER) void savageOUTREG( SavagePtr psav, unsigned long offset, unsigned long value ); static XF86VideoAdaptorPtr SavageSetupImageVideo(ScreenPtr); static void SavageInitOffscreenImages(ScreenPtr); static void SavageStopVideo(ScrnInfoPtr, pointer, Bool); static int SavageSetPortAttribute(ScrnInfoPtr, Atom, INT32, pointer); static int SavageGetPortAttribute(ScrnInfoPtr, Atom ,INT32 *, pointer); static void SavageQueryBestSize(ScrnInfoPtr, Bool, short, short, short, short, unsigned int *, unsigned int *, pointer); static int SavagePutImage( ScrnInfoPtr, short, short, short, short, short, short, short, short, int, unsigned char*, short, short, Bool, RegionPtr, pointer, DrawablePtr); static int SavageQueryImageAttributes(ScrnInfoPtr, int, unsigned short *, unsigned short *, int *, int *); static void SavageFreeMemory(ScrnInfoPtr pScrn, void *mem_struct); void SavageResetVideo(ScrnInfoPtr pScrn); static void SavageSetColorKeyOld(ScrnInfoPtr pScrn); static void SavageSetColorKeyNew(ScrnInfoPtr pScrn); static void SavageSetColorKey2000(ScrnInfoPtr pScrn); static void (*SavageSetColorKey)(ScrnInfoPtr pScrn) = NULL; static void SavageSetColorOld(ScrnInfoPtr pScrn ); static void SavageSetColorNew(ScrnInfoPtr pScrn ); static void SavageSetColor2000(ScrnInfoPtr pScrn ); static void (*SavageSetColor)(ScrnInfoPtr pScrn ) = NULL; static void (*SavageInitStreams)(ScrnInfoPtr pScrn) = NULL; static void SavageDisplayVideoOld( ScrnInfoPtr pScrn, int id, int offset, short width, short height, int pitch, int x1, int y1, int x2, int y2, BoxPtr dstBox, short src_w, short src_h, short drw_w, short drw_h ); static void SavageDisplayVideoNew( ScrnInfoPtr pScrn, int id, int offset, short width, short height, int pitch, int x1, int y1, int x2, int y2, BoxPtr dstBox, short src_w, short src_h, short drw_w, short drw_h ); static void SavageDisplayVideo2000( ScrnInfoPtr pScrn, int id, int offset, short width, short height, int pitch, int x1, int y1, int x2, int y2, BoxPtr dstBox, short src_w, short src_h, short drw_w, short drw_h ); static void (*SavageDisplayVideo)( ScrnInfoPtr pScrn, int id, int offset, short width, short height, int pitch, int x1, int y1, int x2, int y2, BoxPtr dstBox, short src_w, short src_h, short drw_w, short drw_h ) = NULL; /*static void SavageBlockHandler(int, pointer, pointer, pointer);*/ #define XVTRACE 4 #define MAKE_ATOM(a) MakeAtom(a, sizeof(a) - 1, TRUE) static Atom xvColorKey, xvBrightness, xvContrast, xvSaturation, xvHue, xvInterpolation; /* client libraries expect an encoding */ static XF86VideoEncodingRec DummyEncoding[1] = { { 0, "XV_IMAGE", 1024, 1024, {1, 1} } }; #define NUM_FORMATS 5 static XF86VideoFormatRec Formats[NUM_FORMATS] = { {8, PseudoColor}, {15, TrueColor}, {16, TrueColor}, {24, TrueColor} }; #define NUM_ATTRIBUTES 6 static XF86AttributeRec Attributes[NUM_ATTRIBUTES] = { {XvSettable | XvGettable, 0, (1 << 24) - 1, "XV_COLORKEY"}, {XvSettable | XvGettable, -128, 127, "XV_BRIGHTNESS"}, {XvSettable | XvGettable, 0, 255, "XV_CONTRAST"}, {XvSettable | XvGettable, 0, 255, "XV_SATURATION"}, {XvSettable | XvGettable, -180, 180, "XV_HUE"}, {XvSettable | XvGettable, 0, 1, "XV_VERTICAL_INTERPOLATION"} }; #define FOURCC_RV16 0x36315652 #define FOURCC_RV15 0x35315652 #define FOURCC_Y211 0x31313259 /* * For completeness sake, here is a cracking of the fourcc's I support. * * YUY2, packed 4:2:2, byte order: Y0 U0 Y1 V0 Y2 U2 Y3 V2 * Y211, packed 2:1:1, byte order: Y0 U0 Y2 V0 Y4 U2 Y6 V2 * YV12, planar 4:1:1, Y plane HxW, V plane H/2xW/2, U plane H/2xW/2 * I420, planar 4:1:1, Y plane HxW, U plane H/2xW/2, V plane H/2xW/2 * (I420 is also known as IYUV) */ static XF86ImageRec Images[] = { XVIMAGE_YUY2, XVIMAGE_YV12, XVIMAGE_I420, { FOURCC_RV15, XvRGB, LSBFirst, {'R','V','1','5', 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00}, 16, XvPacked, 1, 15, 0x001F, 0x03E0, 0x7C00, 0, 0, 0, 0, 0, 0, 0, 0, 0, {'R','V','B',0, 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0}, XvTopToBottom }, { FOURCC_RV16, XvRGB, LSBFirst, {'R','V','1','6', 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00}, 16, XvPacked, 1, 16, 0x001F, 0x07E0, 0xF800, 0, 0, 0, 0, 0, 0, 0, 0, 0, {'R','V','B',0, 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0}, XvTopToBottom }, { FOURCC_Y211, XvYUV, LSBFirst, {'Y','2','1','1', 0x00,0x00,0x00,0x10,0x80,0x00,0x00,0xAA,0x00,0x38,0x9B,0x71}, 6, XvPacked, 3, 0, 0, 0, 0 , 8, 8, 8, 2, 4, 4, 1, 1, 1, {'Y','U','Y','V', 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0}, XvTopToBottom } }; #define NUM_IMAGES (sizeof(Images)/sizeof(Images[0])) typedef struct { int brightness; /* -128 .. 127 */ CARD32 contrast; /* 0 .. 255 */ CARD32 saturation; /* 0 .. 255 */ int hue; /* -128 .. 127 */ Bool interpolation; /* on/off */ /*FBAreaPtr area;*/ RegionRec clip; CARD32 colorKey; CARD32 videoStatus; Time offTime; Time freeTime; int lastKnownPitch; void *video_memory; /* opaque memory management information structure */ CARD32 video_offset; /* offset in video memory of packed YUV buffer */ void *video_planarmem; /* opaque memory management information structure */ CARD32 video_planarbuf; /* offset in video memory of planar YV12 buffer */ #ifdef SAVAGEDRI Bool tried_agp; /* TRUE if AGP allocation has been tried */ CARD32 agpBase; /* Physical address of aperture base */ CARD32 agpBufferOffset; /* Offset of buffer in AGP memory, or 0 if unavailable */ drmAddress agpBufferMap; /* Mapping of AGP buffer in process memory, or NULL */ #endif } SavagePortPrivRec, *SavagePortPrivPtr; #define GET_PORT_PRIVATE(pScrn) \ (SavagePortPrivPtr)((SAVPTR(pScrn))->adaptor->pPortPrivates[0].ptr) static unsigned int GetBlendForFourCC( int id ) { switch( id ) { case FOURCC_YUY2: case FOURCC_YV12: /* shouldn't this be 4? */ case FOURCC_I420: /* shouldn't this be 4? */ return 1; case FOURCC_Y211: return 4; case FOURCC_RV15: return 3; case FOURCC_RV16: return 5; default: return 0; } } static unsigned int GetBlendForFourCC2000( int id ) { switch( id ) { case FOURCC_YUY2: return 1; case FOURCC_I420: return 1; /* was 4 */ case FOURCC_YV12: return 1; /* was 4 */ case FOURCC_Y211: return 4; case FOURCC_RV15: return 3; case FOURCC_RV16: return 5; default: return 0; } } void savageOUTREG( SavagePtr psav, unsigned long offset, unsigned long value ) { ErrorF( "MMIO %08lx, was %08lx, want %08lx,", offset, (unsigned long)MMIO_IN32( psav->MapBase, offset ), value ); MMIO_OUT32( psav->MapBase, offset, value ); ErrorF( " now %08lx\n", (unsigned long)MMIO_IN32( psav->MapBase, offset ) ); } #if 0 static void SavageClipVWindow(ScrnInfoPtr pScrn) { SavagePtr psav = SAVPTR(pScrn); if( (psav->Chipset == S3_SAVAGE_MX) || (psav->Chipset == S3_SUPERSAVAGE) ) { if (psav->IsSecondary) { OUTREG(SEC_STREAM2_WINDOW_SZ, 0); } else if (psav->IsPrimary) { OUTREG(SEC_STREAM_WINDOW_SZ, 0); } else { OUTREG(SEC_STREAM_WINDOW_SZ, 0); #if 0 OUTREG(SEC_STREAM2_WINDOW_SZ, 0); #endif } } else if (psav->Chipset == S3_SAVAGE2000) { OUTREG(SEC_STREAM_WINDOW_SZ, 0); } else { OUTREG( SSTREAM_WINDOW_SIZE_REG, 1); OUTREG( SSTREAM_WINDOW_START_REG, 0x03ff03ff); } } #endif void SavageInitVideo(ScreenPtr pScreen) { ScrnInfoPtr pScrn = xf86ScreenToScrn(pScreen); XF86VideoAdaptorPtr *adaptors, *newAdaptors = NULL; XF86VideoAdaptorPtr newAdaptor = NULL; SavagePtr psav = SAVPTR(pScrn); int num_adaptors; xf86ErrorFVerb(XVTRACE,"SavageInitVideo\n"); if (S3_SAVAGE_MOBILE_SERIES(psav->Chipset)) { newAdaptor = SavageSetupImageVideo(pScreen); SavageInitOffscreenImages(pScreen); SavageInitStreams = SavageInitStreamsNew; SavageSetColor = SavageSetColorNew; SavageSetColorKey = SavageSetColorKeyNew; SavageDisplayVideo = SavageDisplayVideoNew; } else if (psav->Chipset == S3_SAVAGE2000) { newAdaptor = SavageSetupImageVideo(pScreen); SavageInitOffscreenImages(pScreen); SavageInitStreams = SavageInitStreams2000; SavageSetColor = SavageSetColor2000; SavageSetColorKey = SavageSetColorKey2000; SavageDisplayVideo = SavageDisplayVideo2000; } else { newAdaptor = SavageSetupImageVideo(pScreen); SavageInitOffscreenImages(pScreen); SavageInitStreams = SavageInitStreamsOld; SavageSetColor = SavageSetColorOld; SavageSetColorKey = SavageSetColorKeyOld; SavageDisplayVideo = SavageDisplayVideoOld; } num_adaptors = xf86XVListGenericAdaptors(pScrn, &adaptors); if(newAdaptor) { if(!num_adaptors) { num_adaptors = 1; adaptors = &newAdaptor; } else { newAdaptors = /* need to free this someplace */ malloc((num_adaptors + 1) * sizeof(XF86VideoAdaptorPtr*)); if(newAdaptors) { memcpy(newAdaptors, adaptors, num_adaptors * sizeof(XF86VideoAdaptorPtr)); newAdaptors[num_adaptors] = newAdaptor; adaptors = newAdaptors; num_adaptors++; } } } if(num_adaptors) xf86XVScreenInit(pScreen, adaptors, num_adaptors); if(newAdaptors) free(newAdaptors); if( newAdaptor ) { psav->videoFourCC = 0; } } void SavageSetColorKeyOld(ScrnInfoPtr pScrn) { SavagePtr psav = SAVPTR(pScrn); SavagePortPrivPtr pPriv = psav->adaptor->pPortPrivates[0].ptr; int red, green, blue; /* Here, we reset the colorkey and all the controls. */ red = (pPriv->colorKey & pScrn->mask.red) >> pScrn->offset.red; green = (pPriv->colorKey & pScrn->mask.green) >> pScrn->offset.green; blue = (pPriv->colorKey & pScrn->mask.blue) >> pScrn->offset.blue; if( !pPriv->colorKey ) { OUTREG( COL_CHROMA_KEY_CONTROL_REG, 0 ); OUTREG( CHROMA_KEY_UPPER_BOUND_REG, 0 ); OUTREG( BLEND_CONTROL_REG, 0 ); } else { switch (pScrn->depth) { case 8: OUTREG( COL_CHROMA_KEY_CONTROL_REG, 0x37000000 | (pPriv->colorKey & 0xFF) ); OUTREG( CHROMA_KEY_UPPER_BOUND_REG, 0x00000000 | (pPriv->colorKey & 0xFF) ); break; case 15: OUTREG( COL_CHROMA_KEY_CONTROL_REG, 0x05000000 | (red<<19) | (green<<11) | (blue<<3) ); OUTREG( CHROMA_KEY_UPPER_BOUND_REG, 0x00000000 | (red<<19) | (green<<11) | (blue<<3) ); break; case 16: OUTREG( COL_CHROMA_KEY_CONTROL_REG, 0x16000000 | (red<<19) | (green<<10) | (blue<<3) ); OUTREG( CHROMA_KEY_UPPER_BOUND_REG, 0x00020002 | (red<<19) | (green<<10) | (blue<<3) ); break; case 24: OUTREG( COL_CHROMA_KEY_CONTROL_REG, 0x17000000 | (red<<16) | (green<<8) | (blue) ); OUTREG( CHROMA_KEY_UPPER_BOUND_REG, 0x00000000 | (red<<16) | (green<<8) | (blue) ); break; } /* We use destination colorkey */ OUTREG( BLEND_CONTROL_REG, 0x05000000 ); } } void SavageSetColorKeyNew(ScrnInfoPtr pScrn) { SavagePtr psav = SAVPTR(pScrn); SavagePortPrivPtr pPriv = psav->adaptor->pPortPrivates[0].ptr; int red, green, blue; /* Here, we reset the colorkey and all the controls. */ red = (pPriv->colorKey & pScrn->mask.red) >> pScrn->offset.red; green = (pPriv->colorKey & pScrn->mask.green) >> pScrn->offset.green; blue = (pPriv->colorKey & pScrn->mask.blue) >> pScrn->offset.blue; if( !pPriv->colorKey ) { if (psav->IsSecondary) { OUTREG( SEC_STREAM2_CKEY_LOW, 0 ); OUTREG( SEC_STREAM2_CKEY_UPPER, 0 ); OUTREG( BLEND_CONTROL, (INREG32(BLEND_CONTROL) | (psav->blendBase << 17) | (8 << 12) )); } else if (psav->IsPrimary) { OUTREG( SEC_STREAM_CKEY_LOW, 0 ); OUTREG( SEC_STREAM_CKEY_UPPER, 0 ); OUTREG( BLEND_CONTROL, (INREG32(BLEND_CONTROL) | (psav->blendBase << 9) | 0x08 )); } else { OUTREG( SEC_STREAM_CKEY_LOW, 0 ); OUTREG( SEC_STREAM_CKEY_UPPER, 0 ); OUTREG( BLEND_CONTROL, (INREG32(BLEND_CONTROL) | (psav->blendBase << 9) | 0x08 )); #if 0 sleep(1); OUTREG( SEC_STREAM2_CKEY_LOW, 0 ); OUTREG( SEC_STREAM2_CKEY_UPPER, 0 ); OUTREG( BLEND_CONTROL, (INREG32(BLEND_CONTROL) | (psav->blendBase << 17) | (8 << 12) )); #endif } } else { switch (pScrn->depth) { case 8: if (psav->IsSecondary) { OUTREG( SEC_STREAM2_CKEY_LOW, 0x47000000 | (pPriv->colorKey & 0xFF) ); OUTREG( SEC_STREAM2_CKEY_UPPER, 0x47000000 | (pPriv->colorKey & 0xFF) ); } else if (psav->IsPrimary) { OUTREG( SEC_STREAM_CKEY_LOW, 0x47000000 | (pPriv->colorKey & 0xFF) ); OUTREG( SEC_STREAM_CKEY_UPPER, 0x47000000 | (pPriv->colorKey & 0xFF) ); } else { OUTREG( SEC_STREAM_CKEY_LOW, 0x47000000 | (pPriv->colorKey & 0xFF) ); OUTREG( SEC_STREAM_CKEY_UPPER, 0x47000000 | (pPriv->colorKey & 0xFF) ); #if 0 OUTREG( SEC_STREAM2_CKEY_LOW, 0x47000000 | (pPriv->colorKey & 0xFF) ); OUTREG( SEC_STREAM2_CKEY_UPPER, 0x47000000 | (pPriv->colorKey & 0xFF) ); #endif } break; case 15: if (psav->IsSecondary) { OUTREG( SEC_STREAM2_CKEY_LOW, 0x45000000 | (red<<19) | (green<<11) | (blue<<3) ); OUTREG( SEC_STREAM2_CKEY_UPPER, 0x45000000 | (red<<19) | (green<<11) | (blue<<3) ); } else if (psav->IsPrimary) { OUTREG( SEC_STREAM_CKEY_LOW, 0x45000000 | (red<<19) | (green<<11) | (blue<<3) ); OUTREG( SEC_STREAM_CKEY_UPPER, 0x45000000 | (red<<19) | (green<<11) | (blue<<3) ); } else { OUTREG( SEC_STREAM_CKEY_LOW, 0x45000000 | (red<<19) | (green<<11) | (blue<<3) ); OUTREG( SEC_STREAM_CKEY_UPPER, 0x45000000 | (red<<19) | (green<<11) | (blue<<3) ); #if 0 OUTREG( SEC_STREAM2_CKEY_LOW, 0x45000000 | (red<<19) | (green<<11) | (blue<<3) ); OUTREG( SEC_STREAM2_CKEY_UPPER, 0x45000000 | (red<<19) | (green<<11) | (blue<<3) ); #endif } break; case 16: if (psav->IsSecondary) { OUTREG( SEC_STREAM2_CKEY_LOW, 0x46000000 | (red<<19) | (green<<10) | (blue<<3) ); OUTREG( SEC_STREAM2_CKEY_UPPER, 0x46020002 | (red<<19) | (green<<10) | (blue<<3) ); } else if (psav->IsPrimary) { OUTREG( SEC_STREAM_CKEY_LOW, 0x46000000 | (red<<19) | (green<<10) | (blue<<3) ); OUTREG( SEC_STREAM_CKEY_UPPER, 0x46020002 | (red<<19) | (green<<10) | (blue<<3) ); } else { OUTREG( SEC_STREAM_CKEY_LOW, 0x46000000 | (red<<19) | (green<<10) | (blue<<3) ); OUTREG( SEC_STREAM_CKEY_UPPER, 0x46020002 | (red<<19) | (green<<10) | (blue<<3) ); #if 0 OUTREG( SEC_STREAM2_CKEY_LOW, 0x46000000 | (red<<19) | (green<<10) | (blue<<3) ); OUTREG( SEC_STREAM2_CKEY_UPPER, 0x46020002 | (red<<19) | (green<<10) | (blue<<3) ); #endif } break; case 24: if (psav->IsSecondary) { OUTREG( SEC_STREAM2_CKEY_LOW, 0x47000000 | (red<<16) | (green<<8) | (blue) ); OUTREG( SEC_STREAM2_CKEY_UPPER, 0x47000000 | (red<<16) | (green<<8) | (blue) ); } else if (psav->IsPrimary) { OUTREG( SEC_STREAM_CKEY_LOW, 0x47000000 | (red<<16) | (green<<8) | (blue) ); OUTREG( SEC_STREAM_CKEY_UPPER, 0x47000000 | (red<<16) | (green<<8) | (blue) ); } else { OUTREG( SEC_STREAM_CKEY_LOW, 0x47000000 | (red<<16) | (green<<8) | (blue) ); OUTREG( SEC_STREAM_CKEY_UPPER, 0x47000000 | (red<<16) | (green<<8) | (blue) ); #if 0 OUTREG( SEC_STREAM2_CKEY_LOW, 0x47000000 | (red<<16) | (green<<8) | (blue) ); OUTREG( SEC_STREAM2_CKEY_UPPER, 0x47000000 | (red<<16) | (green<<8) | (blue) ); #endif } break; } /* We assume destination colorkey */ if (psav->IsSecondary) { OUTREG( BLEND_CONTROL, (INREG32(BLEND_CONTROL) | (psav->blendBase << 17) | (8 << 12) )); } else if (psav->IsPrimary) { OUTREG( BLEND_CONTROL, (INREG32(BLEND_CONTROL) | (psav->blendBase << 9) | 0x08 )); } else { OUTREG( BLEND_CONTROL, (INREG32(BLEND_CONTROL) | (psav->blendBase << 9) | 0x08 )); #if 0 OUTREG( BLEND_CONTROL, (INREG32(BLEND_CONTROL) | (psav->blendBase << 17) | (8 << 12) )); #endif } } } void SavageSetColorKey2000(ScrnInfoPtr pScrn) { SavagePtr psav = SAVPTR(pScrn); SavagePortPrivPtr pPriv = psav->adaptor->pPortPrivates[0].ptr; int red, green, blue; /* Here, we reset the colorkey and all the controls. */ red = (pPriv->colorKey & pScrn->mask.red) >> pScrn->offset.red; green = (pPriv->colorKey & pScrn->mask.green) >> pScrn->offset.green; blue = (pPriv->colorKey & pScrn->mask.blue) >> pScrn->offset.blue; if( !pPriv->colorKey ) { OUTREG( SEC_STREAM_CKEY_LOW, 0); OUTREG( SEC_STREAM_CKEY_UPPER, 0); OUTREG( BLEND_CONTROL, (8 << 2)); } else { switch (pScrn->depth) { case 8: OUTREG( SEC_STREAM_CKEY_LOW, 0x47000000 | (pPriv->colorKey & 0xFF) ); OUTREG( SEC_STREAM_CKEY_UPPER, (pPriv->colorKey & 0xFF) ); break; case 15: OUTREG( SEC_STREAM_CKEY_LOW, 0x45000000 | (red<<19) | (green<<11) | (blue<<3) ); OUTREG( SEC_STREAM_CKEY_UPPER, (red<<19) | (green<<11) | (blue<<3) ); break; case 16: OUTREG( SEC_STREAM_CKEY_LOW, 0x46000000 | (red<<19) | (green<<10) | (blue<<3) ); OUTREG( SEC_STREAM_CKEY_UPPER, (red<<19) | (green<<10) | (blue<<3) ); break; case 24: OUTREG( SEC_STREAM_CKEY_LOW, 0x47000000 | (red<<16) | (green<<8) | (blue) ); OUTREG( SEC_STREAM_CKEY_UPPER, (red<<16) | (green<<8) | (blue) ); break; } /* We assume destination colorkey */ OUTREG( BLEND_CONTROL, INREG(BLEND_CONTROL) | (8 << 2)); } } void SavageSetColorOld( ScrnInfoPtr pScrn ) { SavagePtr psav = SAVPTR(pScrn); SavagePortPrivPtr pPriv = psav->adaptor->pPortPrivates[0].ptr; xf86ErrorFVerb(XVTRACE, "bright %d, contrast %d, saturation %d, hue %d\n", pPriv->brightness, (int)pPriv->contrast, (int)pPriv->saturation, pPriv->hue ); if( (psav->videoFourCC == FOURCC_RV15) || (psav->videoFourCC == FOURCC_RV16) ) { OUTREG( COLOR_ADJUSTMENT_REG, 0 ); } else { /* Change 0..255 into 0..15 */ long sat = pPriv->saturation * 16 / 256; double hue = pPriv->hue * 0.017453292; unsigned long hs1 = ((long)(sat * cos(hue))) & 0x1f; unsigned long hs2 = ((long)(sat * sin(hue))) & 0x1f; OUTREG( COLOR_ADJUSTMENT_REG, 0x80008000 | (pPriv->brightness + 128) | ((pPriv->contrast & 0xf8) << (12-7)) | (hs1 << 16) | (hs2 << 24) ); } } void SavageSetColorNew( ScrnInfoPtr pScrn ) { SavagePtr psav = SAVPTR(pScrn); SavagePortPrivPtr pPriv = psav->adaptor->pPortPrivates[0].ptr; /* Brightness/contrast/saturation/hue computations. */ double k, dk1, dk2, dk3, dk4, dk5, dk6, dk7, dkb; int k1, k2, k3, k4, k5, k6, k7, kb; double s = pPriv->saturation / 128.0; double h = pPriv->hue * 0.017453292; unsigned long assembly1, assembly2, assembly3; xf86ErrorFVerb(XVTRACE, "bright %d, contrast %d, saturation %d, hue %d\n", pPriv->brightness, (int)pPriv->contrast, (int)pPriv->saturation, pPriv->hue ); if( psav->videoFourCC == FOURCC_Y211 ) k = 1.0; /* YUV */ else k = 1.14; /* YCrCb */ /* * The S3 documentation must be wrong for k4 and k5. Their default * values, which they hardcode in their Windows driver, have the * opposite sign from the results in the register spec. */ dk1 = k * pPriv->contrast; dk2 = 64.0 * 1.371 * k * s * cos(h); dk3 = -64.0 * 1.371 * k * s * sin(h); dk4 = -128.0 * k * s * (0.698 * cos(h) - 0.336 * sin(h)); dk5 = -128.0 * k * s * (0.698 * sin(h) + 0.336 * cos(h)); dk6 = 64.0 * 1.732 * k * s * sin(h); /* == k3 / 1.26331, right? */ dk7 = 64.0 * 1.732 * k * s * cos(h); /* == k2 / -1.26331, right? */ dkb = 128.0 * pPriv->brightness + 64.0; if( psav->videoFourCC != FOURCC_Y211 ) dkb -= dk1 * 14.0; k1 = (int)(dk1+0.5) & 0x1ff; k2 = (int)(dk2+0.5) & 0x1ff; k3 = (int)(dk3+0.5) & 0x1ff; assembly1 = (k3<<18) | (k2<<9) | k1; xf86ErrorFVerb(XVTRACE+1, "CC1 = %08lx ", assembly1 ); k4 = (int)(dk4+0.5) & 0x1ff; k5 = (int)(dk5+0.5) & 0x1ff; k6 = (int)(dk6+0.5) & 0x1ff; assembly2 = (k6<<18) | (k5<<9) | k4; xf86ErrorFVerb(XVTRACE+1, "CC2 = %08lx ", assembly2 ); k7 = (int)(dk7+0.5) & 0x1ff; kb = (int)(dkb+0.5) & 0xffff; assembly3 = (kb<<9) | k7; xf86ErrorFVerb(XVTRACE+1, "CC3 = %08lx\n", assembly3 ); if (psav->IsSecondary) { OUTREG( SEC_STREAM2_COLOR_CONVERT1, assembly1 ); OUTREG( SEC_STREAM2_COLOR_CONVERT2, assembly2 ); OUTREG( SEC_STREAM2_COLOR_CONVERT3, assembly3 ); } else if (psav->IsPrimary) { OUTREG( SEC_STREAM_COLOR_CONVERT3, assembly1 ); OUTREG( SEC_STREAM_COLOR_CONVERT3, assembly2 ); OUTREG( SEC_STREAM_COLOR_CONVERT3, assembly3 ); } else { OUTREG( SEC_STREAM_COLOR_CONVERT3, assembly1 ); OUTREG( SEC_STREAM_COLOR_CONVERT3, assembly2 ); OUTREG( SEC_STREAM_COLOR_CONVERT3, assembly3 ); #if 0 sleep(1); OUTREG( SEC_STREAM2_COLOR_CONVERT1, assembly1 ); OUTREG( SEC_STREAM2_COLOR_CONVERT2, assembly2 ); OUTREG( SEC_STREAM2_COLOR_CONVERT3, assembly3 ); #endif } } void SavageSetColor2000( ScrnInfoPtr pScrn ) { SavagePtr psav = SAVPTR(pScrn); SavagePortPrivPtr pPriv = psav->adaptor->pPortPrivates[0].ptr; /* Brightness/contrast/saturation/hue computations. */ double k, yb, dk1, dk2, dk3, dk4, dk5, dk6, dk7, dkb; int k1, k2, k3, k4, k5, k6, k7, kb; double s = pPriv->saturation / 10000.0; double h = pPriv->hue * 0.017453292; unsigned long assembly1, assembly2, assembly3, assembly4; unsigned long brightness = pPriv->brightness; xf86ErrorFVerb(XVTRACE, "bright %d, contrast %d, saturation %d, hue %d\n", pPriv->brightness, (int)pPriv->contrast, (int)pPriv->saturation, pPriv->hue ); if( psav->videoFourCC == FOURCC_Y211 ) { k = 1.0;/* YUV */ yb = 0.0; } else { k = 1.1;/* YCrCb */ yb = 14.0; } dk1 = 128 * k * (pPriv->contrast / 10000.0); if (dk1 < 0) dk1 -= 0.5; else dk1 += 0.5; dk2 = 64.0 * 1.371 * k * s * cos(h); if (dk2 < 0) dk2 -= 0.5; else dk2 += 0.5; dk3 = -64.0 * 1.371 * k * s * sin(h); if (dk3 < 0) dk3 -= 0.5; else dk3 += 0.5; dk4 = -128.0 * k * s * (0.698 * cos(h) + 0.336 * sin(h)); if (dk4 < 0) dk4 -= 0.5; else dk4 += 0.5; dk5 = 128.0 * k * s * (0.698 * sin(h) - 0.336 * cos(h)); if (dk5 < 0) dk5 -= 0.5; else dk5 += 0.5; dk6 = 64.0 * 1.732 * k * s * sin(h); if (dk6 < 0) dk6 -= 0.5; else dk6 += 0.5; dk7 = 64.0 * 1.732 * k * s * cos(h); if (dk7 < 0) dk7 -= 0.5; else dk7 += 0.5; if (pPriv->brightness <= 0) brightness = pPriv->brightness * 200 / 750 - 200; else brightness = (pPriv->brightness - 750) * 200 / (10000 - 750); dkb = 128 * (brightness - (k * pPriv->contrast * yb / 10000.0) + 0.5); if (dkb < 0) dkb -= 0.5; else dkb += 0.5; k1 = (int)(dk1 /*+0.5*/) & 0x1ff; k2 = (int)(dk2 /*+0.5*/) & 0x1ff; assembly1 = (k2<<16) | k1; k3 = (int)(dk3 /*+0.5*/) & 0x1ff; k4 = (int)(dk4 /*+0.5*/) & 0x1ff; assembly2 = (k4<<16) | k3; k5 = (int)(dk5 /*+0.5*/) & 0x1ff; k6 = (int)(dk6 /*+0.5*/) & 0x1ff; assembly3 = (k6<<16) | k5; k7 = (int)(dk7 /*+0.5*/) & 0x1ff; kb = (int)(dkb /*+0.5*/) & 0xffff; assembly4 = (kb<<16) | k7; #if 0 assembly1 = 0x640092; assembly2 = 0x19a0000; assembly3 = 0x001cf; assembly4 = 0xf8ca007e; #endif OUTREG( SEC_STREAM_COLOR_CONVERT0_2000, assembly1 ); OUTREG( SEC_STREAM_COLOR_CONVERT1_2000, assembly2 ); OUTREG( SEC_STREAM_COLOR_CONVERT2_2000, assembly3 ); OUTREG( SEC_STREAM_COLOR_CONVERT3_2000, assembly4 ); } void SavageResetVideo(ScrnInfoPtr pScrn) { xf86ErrorFVerb(XVTRACE,"SavageResetVideo\n"); SavageSetColor( pScrn ); SavageSetColorKey( pScrn ); } static XF86VideoAdaptorPtr SavageSetupImageVideo(ScreenPtr pScreen) { ScrnInfoPtr pScrn = xf86ScreenToScrn(pScreen); SavagePtr psav = SAVPTR(pScrn); XF86VideoAdaptorPtr adapt; SavagePortPrivPtr pPriv; xf86ErrorFVerb(XVTRACE,"SavageSetupImageVideo\n"); if(!(adapt = calloc(1, sizeof(XF86VideoAdaptorRec) + sizeof(SavagePortPrivRec) + sizeof(DevUnion)))) return NULL; adapt->type = XvWindowMask | XvInputMask | XvImageMask; adapt->flags = VIDEO_OVERLAID_IMAGES | VIDEO_CLIP_TO_VIEWPORT; adapt->name = "Savage Streams Engine"; adapt->nEncodings = 1; adapt->pEncodings = DummyEncoding; adapt->nFormats = NUM_FORMATS; adapt->pFormats = Formats; adapt->nPorts = 1; adapt->pPortPrivates = (DevUnion*)(&adapt[1]); pPriv = (SavagePortPrivPtr)(&adapt->pPortPrivates[1]); adapt->pPortPrivates[0].ptr = (pointer)(pPriv); adapt->pAttributes = Attributes; adapt->nImages = NUM_IMAGES; adapt->nAttributes = NUM_ATTRIBUTES; adapt->pImages = Images; adapt->PutVideo = NULL; adapt->PutStill = NULL; adapt->GetVideo = NULL; adapt->GetStill = NULL; adapt->StopVideo = SavageStopVideo; adapt->SetPortAttribute = SavageSetPortAttribute; adapt->GetPortAttribute = SavageGetPortAttribute; adapt->QueryBestSize = SavageQueryBestSize; adapt->PutImage = SavagePutImage; adapt->QueryImageAttributes = SavageQueryImageAttributes; xvBrightness = MAKE_ATOM("XV_BRIGHTNESS"); xvContrast = MAKE_ATOM("XV_CONTRAST"); xvColorKey = MAKE_ATOM("XV_COLORKEY"); xvHue = MAKE_ATOM("XV_HUE"); xvSaturation = MAKE_ATOM("XV_SATURATION"); /* interpolation option only available on "old" streams */ xvInterpolation = MAKE_ATOM("XV_VERTICAL_INTERPOLATION"); pPriv->colorKey = (1 << pScrn->offset.red) | (1 << pScrn->offset.green) | (((pScrn->mask.blue >> pScrn->offset.blue) - 1) << pScrn->offset.blue); pPriv->videoStatus = 0; pPriv->brightness = 0; pPriv->contrast = 128; pPriv->saturation = 128; #if 0 /* * The S3 driver has these values for some of the chips. I have yet * to find any Savage where these make sense. */ pPriv->brightness = 64; pPriv->contrast = 16; pPriv->saturation = 128; #endif pPriv->hue = 0; pPriv->lastKnownPitch = 0; pPriv->interpolation = FALSE; /* gotta uninit this someplace */ REGION_NULL(pScreen, &pPriv->clip); psav->adaptor = adapt; #if 0 psav->BlockHandler = pScreen->BlockHandler; pScreen->BlockHandler = SavageBlockHandler; #endif return adapt; } /* SavageClipVideo - Takes the dst box in standard X BoxRec form (top and left edges inclusive, bottom and right exclusive). The new dst box is returned. The source boundaries are given (x1, y1 inclusive, x2, y2 exclusive) and returned are the new source boundaries in 16.16 fixed point. */ static void SavageClipVideo( BoxPtr dst, INT32 *x1, INT32 *x2, INT32 *y1, INT32 *y2, BoxPtr extents, /* extents of the clip region */ INT32 width, INT32 height ){ INT32 vscale, hscale, delta; int diff; hscale = ((*x2 - *x1) << 16) / (dst->x2 - dst->x1); vscale = ((*y2 - *y1) << 16) / (dst->y2 - dst->y1); *x1 <<= 16; *x2 <<= 16; *y1 <<= 16; *y2 <<= 16; diff = extents->x1 - dst->x1; if(diff > 0) { dst->x1 = extents->x1; *x1 += diff * hscale; } diff = dst->x2 - extents->x2; if(diff > 0) { dst->x2 = extents->x2; *x2 -= diff * hscale; } diff = extents->y1 - dst->y1; if(diff > 0) { dst->y1 = extents->y1; *y1 += diff * vscale; } diff = dst->y2 - extents->y2; if(diff > 0) { dst->y2 = extents->y2; *y2 -= diff * vscale; } if(*x1 < 0) { diff = (- *x1 + hscale - 1)/ hscale; dst->x1 += diff; *x1 += diff * hscale; } delta = *x2 - (width << 16); if(delta > 0) { diff = (delta + hscale - 1)/ hscale; dst->x2 -= diff; *x2 -= diff * hscale; } if(*y1 < 0) { diff = (- *y1 + vscale - 1)/ vscale; dst->y1 += diff; *y1 += diff * vscale; } delta = *y2 - (height << 16); if(delta > 0) { diff = (delta + vscale - 1)/ vscale; dst->y2 -= diff; *y2 -= diff * vscale; } } static void SavageStopVideo(ScrnInfoPtr pScrn, pointer data, Bool shutdown) { SavagePortPrivPtr pPriv = (SavagePortPrivPtr)data; #ifdef SAVAGEDRI SavagePtr psav = SAVPTR(pScrn); #endif xf86ErrorFVerb(XVTRACE,"SavageStopVideo\n"); REGION_EMPTY(pScrn->pScreen, &pPriv->clip); if(shutdown) { /*SavageClipVWindow(pScrn);*/ SavageStreamsOff( pScrn ); #ifdef SAVAGEDRI if (pPriv->agpBufferMap != NULL) { SAVAGEDRIServerPrivatePtr pSAVAGEDRIServer = psav->DRIServerInfo; /* agpXVideo is reused to implement UploadToScreen in EXA */ if (!psav->useEXA) { drmUnmap(pPriv->agpBufferMap, pSAVAGEDRIServer->agpXVideo.size); pSAVAGEDRIServer->agpXVideo.map = NULL; } pPriv->agpBufferMap = NULL; pPriv->agpBufferOffset = 0; } pPriv->tried_agp = FALSE; #endif if (pPriv->video_memory != NULL) { SavageFreeMemory(pScrn, pPriv->video_memory); pPriv->video_memory = NULL; } if (pPriv->video_planarmem != NULL) { SavageFreeMemory(pScrn, pPriv->video_planarmem); pPriv->video_planarmem = NULL; } pPriv->videoStatus = 0; } else { if(pPriv->videoStatus & CLIENT_VIDEO_ON) { pPriv->videoStatus |= OFF_TIMER; pPriv->offTime = currentTime.milliseconds + OFF_DELAY; } } } static int SavageSetPortAttribute( ScrnInfoPtr pScrn, Atom attribute, INT32 value, pointer data ){ SavagePortPrivPtr pPriv = (SavagePortPrivPtr)data; SavagePtr psav = SAVPTR(pScrn); if(attribute == xvColorKey) { pPriv->colorKey = value; if( psav->videoFlags & VF_STREAMS_ON) SavageSetColorKey( pScrn ); REGION_EMPTY(pScrn->pScreen, &pPriv->clip); } else if( attribute == xvBrightness) { if((value < -128) || (value > 127)) return BadValue; pPriv->brightness = value; if( psav->videoFlags & VF_STREAMS_ON) SavageSetColor( pScrn ); } else if( attribute == xvContrast) { if((value < 0) || (value > 255)) return BadValue; pPriv->contrast = value; if( psav->videoFlags & VF_STREAMS_ON) SavageSetColor( pScrn ); } else if( attribute == xvSaturation) { if((value < 0) || (value > 255)) return BadValue; pPriv->saturation = value; if( psav->videoFlags & VF_STREAMS_ON) SavageSetColor( pScrn ); } else if( attribute == xvHue) { if((value < -180) || (value > 180)) return BadValue; pPriv->hue = value; if( psav->videoFlags & VF_STREAMS_ON) SavageSetColor( pScrn ); } else if( attribute == xvInterpolation) { if((value < 0) || (value > 1)) return BadValue; if (value == 1) pPriv->interpolation = TRUE; else pPriv->interpolation = FALSE; } else return BadMatch; return Success; } static int SavageGetPortAttribute( ScrnInfoPtr pScrn, Atom attribute, INT32 *value, pointer data ){ SavagePortPrivPtr pPriv = (SavagePortPrivPtr)data; if(attribute == xvColorKey) { *value = pPriv->colorKey; } else if( attribute == xvBrightness ) { *value = pPriv->brightness; } else if( attribute == xvContrast ) { *value = pPriv->contrast; } else if( attribute == xvHue ) { *value = pPriv->hue; } else if( attribute == xvSaturation ) { *value = pPriv->saturation; } else if( attribute == xvInterpolation ) { *value = pPriv->interpolation; } else return BadMatch; return Success; } static void SavageQueryBestSize( ScrnInfoPtr pScrn, Bool motion, short vid_w, short vid_h, short drw_w, short drw_h, unsigned int *p_w, unsigned int *p_h, pointer data ){ /* What are the real limits for the Savage? */ *p_w = drw_w; *p_h = drw_h; if(*p_w > 16384) *p_w = 16384; } /* SavageCopyPlanarDataBCI() causes artifacts on the screen when used on savage4. * It's probably something with the BCI. Maybe we need a waitforidle() or * something... */ static void SavageCopyPlanarDataBCI( ScrnInfoPtr pScrn, unsigned char *srcY, /* Y */ unsigned char *srcV, /* V */ unsigned char *srcU, /* U */ unsigned char *dst, unsigned char * planarPtr, unsigned long planarOffset, int srcPitch, int srcPitch2, int dstPitch, int h,int w, Bool isAGP) { SavagePtr psav = SAVPTR(pScrn); /* for pixel transfer */ unsigned long offsetY = planarOffset; unsigned long offsetV = offsetY + srcPitch * h; unsigned long offsetU = offsetV + srcPitch2 * (h>>1); unsigned long dstOffset = (unsigned long)dst - (unsigned long)psav->FBBase; unsigned char memType; BCI_GET_PTR; /* copy Y planar */ memcpy(planarPtr, srcY, srcPitch * h); /* copy V planar */ planarPtr = planarPtr + srcPitch * h; memcpy(planarPtr, srcV, srcPitch2 * (h>>1)); /* copy U planar */ planarPtr = planarPtr + srcPitch2 * (h>>1); memcpy(planarPtr, srcU, srcPitch2 * (h>>1)); memType = isAGP ? 3 : 0; /* * Transfer pixel data from one memory location to another location * and reformat the data during the transfer * a. program BCI51 to specify the source information * b. program BCI52 to specify the destination information * c. program BCI53 to specify the source dimensions * d. program BCI54 to specify the destination dimensions * e. (if the data is in YCbCr420 format)program BCI55,BCI56,BCI57 to * locations of the Y,Cb,and Cr data * f. program BCI50(command=011) to specify the formatting options and * kick off the transfer * this command can be used for color space conversion(YCbCr to RGB) * or for oversampling, but not for both simultaneously. it can also be * used to do mastered image transfer when the source is tiled */ w = (w+0xf)&0xff0; psav->WaitQueue(psav,11); BCI_SEND(BCI_SET_REGISTER | BCI_SET_REGISTER_COUNT(7) | 0x51); BCI_SEND(offsetY | memType); BCI_SEND(dstOffset); BCI_SEND(((h-1)<<16)|((w-1)>>3)); BCI_SEND(dstPitch >> 3); BCI_SEND(offsetU | memType); BCI_SEND(offsetV | memType); BCI_SEND((srcPitch2 << 16)| srcPitch2); BCI_SEND(BCI_SET_REGISTER | BCI_SET_REGISTER_COUNT(1) | 0x50); BCI_SEND(0x00200003 | srcPitch); BCI_SEND(0xC0170000); } static void SavageCopyData( unsigned char *src, unsigned char *dst, int srcPitch, int dstPitch, int h, int w ){ w <<= 1; if (w == srcPitch && w == dstPitch) { memcpy(dst, src, w * h); } else while(h--) { memcpy(dst, src, w); src += srcPitch; dst += dstPitch; } } static void SavageCopyPlanarData( unsigned char *src1, /* Y */ unsigned char *src2, /* V */ unsigned char *src3, /* U */ unsigned char *dst1, int srcPitch, int srcPitch2, int dstPitch, int h, int w ){ CARD32 *dst = (CARD32*)dst1; int i, j; dstPitch >>= 2; w >>= 1; for(j = 0; j < h; j++) { for(i = 0; i < w; i++) { /* Shouldn't this be 'if LITTLEENDIAN'? */ #if 1 dst[i] = src1[i << 1] | (src1[(i << 1) + 1] << 16) | (src3[i] << 8) | (src2[i] << 24); #else dst[i] = (src1[i << 1] << 24) | (src1[(i << 1) + 1] << 8) | (src3[i] << 0) | (src2[i] << 16); #endif } dst += dstPitch; src1 += srcPitch; if(j & 1) { src2 += srcPitch2; src3 += srcPitch2; } } } static void SavageVideoSave(ScreenPtr pScreen, ExaOffscreenArea *area) { ScrnInfoPtr pScrn = xf86ScreenToScrn(pScreen); SavagePtr psav = SAVPTR(pScrn); SavagePortPrivPtr pPriv = psav->adaptor->pPortPrivates[0].ptr; if (pPriv->video_memory == area) pPriv->video_memory = NULL; if (pPriv->video_planarmem == area) pPriv->video_planarmem = NULL; } static CARD32 SavageAllocateMemory( ScrnInfoPtr pScrn, void **mem_struct, int size ){ ScreenPtr pScreen = xf86ScrnToScreen(pScrn); SavagePtr psav = SAVPTR(pScrn); int offset = 0; if (psav->useEXA) { ExaOffscreenArea *area = *mem_struct; if (area != NULL) { if (area->size >= size) return area->offset; exaOffscreenFree(pScrn->pScreen, area); } area = exaOffscreenAlloc(pScrn->pScreen, size, 64, TRUE, SavageVideoSave, NULL); *mem_struct = area; if (area == NULL) return 0; offset = area->offset; } if (!psav->useEXA) { FBLinearPtr linear = *mem_struct; int cpp = pScrn->bitsPerPixel / 8; /* XAA allocates in units of pixels at the screen bpp, so adjust size * appropriately. */ size = (size + cpp - 1) / cpp; if (linear) { if(linear->size >= size) return linear->offset * cpp; if(xf86ResizeOffscreenLinear(linear, size)) return linear->offset * cpp; xf86FreeOffscreenLinear(linear); } linear = xf86AllocateOffscreenLinear(pScreen, size, 16, NULL, NULL, NULL); *mem_struct = linear; if (!linear) { int max_size; xf86QueryLargestOffscreenLinear(pScreen, &max_size, 16, PRIORITY_EXTREME); if(max_size < size) return 0; xf86PurgeUnlockedOffscreenAreas(pScreen); linear = xf86AllocateOffscreenLinear(pScreen, size, 16, NULL, NULL, NULL); *mem_struct = linear; if (!linear) return 0; } offset = linear->offset * cpp; } return offset; } static void SavageFreeMemory( ScrnInfoPtr pScrn, void *mem_struct ){ SavagePtr psav = SAVPTR(pScrn); if (psav->useEXA) { ExaOffscreenArea *area = mem_struct; if (area != NULL) exaOffscreenFree(pScrn->pScreen, area); } if (!psav->useEXA) { FBLinearPtr linear = mem_struct; if (linear != NULL) xf86FreeOffscreenLinear(linear); } } static void SavageSetBlend(ScrnInfoPtr pScrn, int id) { SavagePtr psav = SAVPTR(pScrn); if ( S3_SAVAGE_MOBILE_SERIES(psav->Chipset) ) { psav->blendBase = GetBlendForFourCC( id ); xf86ErrorFVerb(XVTRACE+1,"Format %4.4s, blend is %08x\n", (char*)&id, psav->blendBase ); if (psav->IsSecondary) { OUTREG( BLEND_CONTROL, (INREG32(BLEND_CONTROL) | (psav->blendBase << 17) | (8 << 12) )); } else if (psav->IsPrimary) { OUTREG( BLEND_CONTROL, (INREG32(BLEND_CONTROL) | (psav->blendBase << 9) | 0x08 )); } else { OUTREG( BLEND_CONTROL, (INREG32(BLEND_CONTROL) | (psav->blendBase << 9) | 0x08 )); #if 0 OUTREG( BLEND_CONTROL, (INREG32(BLEND_CONTROL) | (psav->blendBase << 17) | (8 << 12) )); #endif } } else if (psav->Chipset == S3_SAVAGE2000) { psav->blendBase = GetBlendForFourCC2000( id ); xf86ErrorFVerb(XVTRACE+1,"Format %4.4s, blend is %08x\n", (char*)&id, psav->blendBase ); if (id != FOURCC_YV12) OUTREG( BLEND_CONTROL, ((psav->blendBase << 24) | (8 << 2) /*| 0x20000000*/)); else OUTREG( BLEND_CONTROL, ((psav->blendBase << 24) | (8 << 2) /*| 0x10000000*/)); } psav->videoFourCC = id; } static void SavageDisplayVideoOld( ScrnInfoPtr pScrn, int id, int offset, short width, short height, int pitch, int x1, int y1, int x2, int y2, BoxPtr dstBox, short src_w, short src_h, short drw_w, short drw_h ){ SavagePtr psav = SAVPTR(pScrn); vgaHWPtr hwp = VGAHWPTR(pScrn); SavagePortPrivPtr pPriv = psav->adaptor->pPortPrivates[0].ptr; /*DisplayModePtr mode = pScrn->currentMode;*/ int vgaCRIndex, vgaCRReg, vgaIOBase; CARD32 ssControl; int scalratio; vgaIOBase = hwp->IOBase; vgaCRIndex = vgaIOBase + 4; vgaCRReg = vgaIOBase + 5; #if 0 if ( psav->videoFourCC != id ) { SavageSetBlend(pScrn,id); SavageResetVideo(pScrn); } #endif if( psav->videoFourCC != id ) SavageStreamsOff(pScrn); if( !(psav->videoFlags & VF_STREAMS_ON) ) { SavageSetBlend(pScrn,id); SavageStreamsOn(pScrn); SavageResetVideo(pScrn); pPriv->lastKnownPitch = 0; } if (S3_MOBILE_TWISTER_SERIES(psav->Chipset) && psav->FPExpansion) { drw_w = (drw_w * psav->XExp1) / psav->XExp2 + 1; drw_h = (drw_h * psav->YExp1) / psav->YExp2 + 1; dstBox->x1 = (dstBox->x1 * psav->XExp1) / psav->XExp2; dstBox->y1 = (dstBox->y1 * psav->YExp1) / psav->YExp2; dstBox->x2 = (dstBox->x2 * psav->XExp1) / psav->XExp2; dstBox->y2 = (dstBox->y2 * psav->YExp1) / psav->YExp2; dstBox->x1 += psav->displayXoffset; dstBox->y1 += psav->displayYoffset; dstBox->x2 += psav->displayXoffset; dstBox->y2 += psav->displayYoffset; } /* * Process horizontal scaling * upscaling and downscaling smaller than 2:1 controlled by MM8198 * MM8190 controls downscaling mode larger than 2:1 * Together MM8190 and MM8198 can set arbitrary downscale up to 64:1 */ scalratio = 0; ssControl = 0; if (src_w >= (drw_w * 2)) { if (src_w < (drw_w * 4)) { ssControl |= HDSCALE_4; scalratio = HSCALING(src_w,(drw_w*4)); } else if (src_w < (drw_w * 8)) { ssControl |= HDSCALE_8; scalratio = HSCALING(src_w,(drw_w*8)); } else if (src_w < (drw_w * 16)) { ssControl |= HDSCALE_16; scalratio = HSCALING(src_w,(drw_w*16)); } else if (src_w < (drw_w * 32)) { ssControl |= HDSCALE_32; scalratio = HSCALING(src_w,(drw_w*32)); } else if (src_w < (drw_w * 64)) { ssControl |= HDSCALE_64; scalratio = HSCALING(src_w,(drw_w*64)); } else { /* Request beyond maximum downscale! */ ssControl |= HDSCALE_64; scalratio = HSCALING(2,1); } } else scalratio = HSCALING(src_w,drw_w); ssControl |= src_w; /*ssControl |= (1 << 24);*/ ssControl |= (GetBlendForFourCC(psav->videoFourCC) << 24); #if 0 /* Wait for VBLANK. */ VerticalRetraceWait(); #endif OUTREG(SSTREAM_CONTROL_REG, ssControl); if (scalratio) OUTREG(SSTREAM_STRETCH_REG,scalratio); /* Calculate vertical scale factor. */ OUTREG(SSTREAM_VINITIAL_REG, 0 ); /*OUTREG(SSTREAM_VSCALE_REG, (src_h << 15) / drw_h );*/ OUTREG(SSTREAM_VSCALE_REG, VSCALING(src_h,drw_h)); /* Set surface location and stride. */ OUTREG(SSTREAM_FBADDR0_REG, (offset + (x1>>15)) & (0x1ffffff & ~BASE_PAD) ); OUTREG(SSTREAM_FBADDR1_REG, 0); OUTREG(SSTREAM_STRIDE_REG, pitch & 0xfff ); OUTREG(SSTREAM_WINDOW_START_REG, OS_XY(dstBox->x1, dstBox->y1) ); OUTREG(SSTREAM_WINDOW_SIZE_REG, OS_WH(dstBox->x2-dstBox->x1, dstBox->y2-dstBox->y1)); /* * MM81E8:Secondary Stream Source Line Count * bit_0~10: # of lines in the source image (before scaling) * bit_15 = 1: Enable vertical interpolation * 0: Line duplicaion */ /* * Vertical Interpolation is very bandwidth intensive. Some savages can't * seem to handle it. Default is line doubling. --AGD */ if (pPriv->interpolation) { if (src_w * 16 <= 0x3300) { OUTREG(SSTREAM_LINES_REG, 0x8000 | src_h ); OUTREG(FIFO_CONTROL, (INREG(FIFO_CONTROL) + 1)); } else { OUTREG(SSTREAM_LINES_REG, src_h ); } } else { OUTREG(SSTREAM_LINES_REG, src_h ); } #if 0 /* Set color key on primary. */ SavageSetColorKey( pScrn ); #endif /* Set FIFO L2 on second stream. */ if( pPriv->lastKnownPitch != pitch ) { unsigned char cr92; pPriv->lastKnownPitch = pitch; pitch = (pitch + 7) / 8; VGAOUT8(vgaCRIndex, 0x92); cr92 = VGAIN8(vgaCRReg); VGAOUT8(vgaCRReg, (cr92 & 0x40) | (pitch >> 8) | 0x80); VGAOUT8(vgaCRIndex, 0x93); if (psav->bTiled && (( drw_h > src_h) || (drw_w > src_w))) VGAOUT8(vgaCRReg, pitch | 0xf); else VGAOUT8(vgaCRReg, pitch); } } static void SavageDisplayVideoNew( ScrnInfoPtr pScrn, int id, int offset, short width, short height, int pitch, int x1, int y1, int x2, int y2, BoxPtr dstBox, short src_w, short src_h, short drw_w, short drw_h ){ SavagePtr psav = SAVPTR(pScrn); vgaHWPtr hwp = VGAHWPTR(pScrn); /*DisplayModePtr mode = pScrn->currentMode;*/ SavagePortPrivPtr pPriv = psav->adaptor->pPortPrivates[0].ptr; int vgaCRIndex, vgaCRReg, vgaIOBase; vgaIOBase = hwp->IOBase; vgaCRIndex = vgaIOBase + 4; vgaCRReg = vgaIOBase + 5; #if 0 if ( psav->videoFourCC != id ) { SavageSetBlend(pScrn,id); SavageResetVideo(pScrn); } #endif if( psav->videoFourCC != id ) SavageStreamsOff(pScrn); if( !(psav->videoFlags & VF_STREAMS_ON) ) { SavageSetBlend(pScrn,id); SavageStreamsOn(pScrn); SavageResetVideo(pScrn); pPriv->lastKnownPitch = 0; } /* Calculate horizontal and vertical scale factors. */ if ( S3_SAVAGE_MOBILE_SERIES(psav->Chipset) && (psav->DisplayType == MT_LCD) && !psav->CrtOnly && !psav->TvOn) { drw_w = (drw_w * psav->XExp1) / psav->XExp2 + 1; drw_h = (drw_h * psav->YExp1) / psav->YExp2 + 1; dstBox->x1 = (dstBox->x1 * psav->XExp1) / psav->XExp2; dstBox->y1 = (dstBox->y1 * psav->YExp1) / psav->YExp2; dstBox->x2 = (dstBox->x2 * psav->XExp1) / psav->XExp2; dstBox->y2 = (dstBox->y2 * psav->YExp1) / psav->YExp2; dstBox->x1 += psav->displayXoffset; dstBox->y1 += psav->displayYoffset; dstBox->x2 += psav->displayXoffset; dstBox->y2 += psav->displayYoffset; } if (psav->IsSecondary) { OUTREG(SEC_STREAM2_HSCALING, ((src_w&0xfff)<<20) | ((65536 * src_w / drw_w) & 0x1FFFF )); /* BUGBUG need to add 00040000 if src stride > 2048 */ OUTREG(SEC_STREAM2_VSCALING, ((src_h&0xfff)<<20) | ((65536 * src_h / drw_h) & 0x1FFFF )); } else if (psav->IsPrimary) { OUTREG(SEC_STREAM_HSCALING, ((src_w&0xfff)<<20) | ((65536 * src_w / drw_w) & 0x1FFFF )); /* BUGBUG need to add 00040000 if src stride > 2048 */ OUTREG(SEC_STREAM_VSCALING, ((src_h&0xfff)<<20) | ((65536 * src_h / drw_h) & 0x1FFFF )); } else { OUTREG(SEC_STREAM_HSCALING, ((src_w&0xfff)<<20) | ((65536 * src_w / drw_w) & 0x1FFFF )); /* BUGBUG need to add 00040000 if src stride > 2048 */ OUTREG(SEC_STREAM_VSCALING, ((src_h&0xfff)<<20) | ((65536 * src_h / drw_h) & 0x1FFFF )); #if 0 OUTREG(SEC_STREAM2_HSCALING, ((src_w&0xfff)<<20) | ((65536 * src_w / drw_w) & 0x1FFFF )); /* BUGBUG need to add 00040000 if src stride > 2048 */ OUTREG(SEC_STREAM2_VSCALING, ((src_h&0xfff)<<20) | ((65536 * src_h / drw_h) & 0x1FFFF )); #endif } /* * Set surface location and stride. We use x1>>15 because all surfaces * are 2 bytes/pixel. */ if (psav->IsSecondary) { OUTREG(SEC_STREAM2_FBUF_ADDR0, (offset + (x1>>15)) & (0x7ffffff & ~BASE_PAD)); OUTREG(SEC_STREAM2_STRIDE_LPB, pitch & 0xfff ); OUTREG(SEC_STREAM2_WINDOW_START, ((dstBox->x1+1) << 16) | (dstBox->y1+1) ); OUTREG(SEC_STREAM2_WINDOW_SZ, ((dstBox->x2-dstBox->x1) << 16) | (dstBox->y2 - dstBox->y1) ); } else if (psav->IsPrimary) { OUTREG(SEC_STREAM_FBUF_ADDR0, (offset + (x1>>15)) & (0x7ffffff & ~BASE_PAD)); OUTREG(SEC_STREAM_STRIDE, pitch & 0xfff ); OUTREG(SEC_STREAM_WINDOW_START, ((dstBox->x1+1) << 16) | (dstBox->y1+1) ); OUTREG(SEC_STREAM_WINDOW_SZ, ((dstBox->x2-dstBox->x1) << 16) | (dstBox->y2 - dstBox->y1) ); } else { OUTREG(SEC_STREAM_FBUF_ADDR0, (offset + (x1>>15)) & (0x7ffffff & ~BASE_PAD)); OUTREG(SEC_STREAM_STRIDE, pitch & 0xfff ); OUTREG(SEC_STREAM_WINDOW_START, ((dstBox->x1+1) << 16) | (dstBox->y1+1) ); OUTREG(SEC_STREAM_WINDOW_SZ, ((dstBox->x2-dstBox->x1) << 16) | (dstBox->y2 - dstBox->y1) ); #if 0 OUTREG(SEC_STREAM2_FBUF_ADDR0, (offset + (x1>>15)) & (0x7ffffff & ~BASE_PAD)); OUTREG(SEC_STREAM2_STRIDE_LPB, pitch & 0xfff ); OUTREG(SEC_STREAM2_WINDOW_START, ((dstBox->x1+1) << 16) | (dstBox->y1+1) ); OUTREG(SEC_STREAM2_WINDOW_SZ, ((dstBox->x2-dstBox->x1) << 16) | (dstBox->y2 - dstBox->y1) ); #endif } #if 0 /* Set color key on primary. */ SavageSetColorKey( pScrn ); #endif /* Set FIFO L2 on second stream. */ /* Is CR92 shadowed for crtc2? -- AGD */ if( pPriv->lastKnownPitch != pitch ) { unsigned char cr92; pPriv->lastKnownPitch = pitch; pitch = (pitch + 7) / 8 - 4; VGAOUT8(vgaCRIndex, 0x92); cr92 = VGAIN8(vgaCRReg); VGAOUT8(vgaCRReg, (cr92 & 0x40) | (pitch >> 8) | 0x80); VGAOUT8(vgaCRIndex, 0x93); VGAOUT8(vgaCRReg, pitch); } } static void SavageDisplayVideo2000( ScrnInfoPtr pScrn, int id, int offset, short width, short height, int pitch, int x1, int y1, int x2, int y2, BoxPtr dstBox, short src_w, short src_h, short drw_w, short drw_h ){ SavagePtr psav = SAVPTR(pScrn); #if 0 vgaHWPtr hwp = VGAHWPTR(pScrn); #endif /*DisplayModePtr mode = pScrn->currentMode;*/ SavagePortPrivPtr pPriv = psav->adaptor->pPortPrivates[0].ptr; #if 0 int vgaCRIndex, vgaCRReg, vgaIOBase; #endif CARD32 addr0; #if 0 CARD32 addr1, addr2; #endif #if 0 vgaIOBase = hwp->IOBase; vgaCRIndex = vgaIOBase + 4; vgaCRReg = vgaIOBase + 5; #endif if( psav->videoFourCC != id ) SavageStreamsOff(pScrn); if( !(psav->videoFlags & VF_STREAMS_ON) ) { SavageSetBlend(pScrn,id); SavageStreamsOn(pScrn); SavageResetVideo(pScrn); pPriv->lastKnownPitch = 0; } if (src_w > drw_w) OUTREG(SEC_STREAM_SRC_START_2000, 0); else OUTREG(SEC_STREAM_SRC_START_2000, SRCSTART(x1, y1)); /*OUTREG(SEC_STREAM_SRC_SIZE_2000, SRCSIZE(src_w, src_h));*/ OUTREG(SEC_STREAM_SRC_SIZE_2000, SRCSIZE((dstBox->x2-dstBox->x1), (dstBox->y2-dstBox->y1))); /* buffersize = (src_w * src_h * 2) / 4096; OUTREG(SEC_STREAM_BUFFERSIZE_2000, (buffersize & 0xffffff) << 12); */ /*SavageResetVideo(pScrn);*/ if( src_w > drw_w ) OUTREG(SEC_STREAM_HSCALE_NORMALIZE, HSCALING_NORMALIZE(drw_w,src_w)); else OUTREG(SEC_STREAM_HSCALE_NORMALIZE, (2048 << 16)); /* Calculate horizontal and vertical scale factors. */ if ((src_w > drw_w) || (src_h > drw_h)) OUTREG(SEC_STREAM_HSCALING, (HSCALING_2000(src_w,drw_w)) | 0x01000000); else OUTREG(SEC_STREAM_HSCALING, HSCALING_2000(src_w,drw_w)); OUTREG(SEC_STREAM_VSCALING, VSCALING_2000(src_h,drw_h)); /* * Set surface location and stride. We use x1>>15 because all surfaces * are 2 bytes/pixel. */ addr0 = offset + (x1>>15); /* Y in YCbCr420 */ #if 0 addr1 = addr0 + (width * height); /* Cb in in YCbCr420 */ addr2 = addr1 + ((width * height) / 4); /* Cr in in YCbCr420 */ #endif OUTREG(SEC_STREAM_FBUF_ADDR0, (addr0) & (0x3fffff & ~BASE_PAD)); #if 0 OUTREG(SEC_STREAM_FBUF_ADDR1, (addr1) & (0x3fffff & ~BASE_PAD)); OUTREG(SEC_STREAM_FBUF_ADDR2, (addr2) & (0x3fffff & ~BASE_PAD)); #endif OUTREG(SEC_STREAM_WINDOW_START, XY_2000(dstBox->x1,dstBox->y1)); OUTREG(SEC_STREAM_WINDOW_SZ, WH_2000((dstBox->x2-dstBox->x1),(dstBox->y2-dstBox->y1))); /*pitch = width * 2;*/ OUTREG(SEC_STREAM_STRIDE, pitch & 0xfff); #if 0 /* Y stride + CbCr stride in YCbCr420 */ OUTREG(SEC_STREAM_STRIDE, (pitch & 0xfff) + ((pitch & 0xfff) << 15)); #endif #if 0 /* Set color key on primary. */ SavageSetColorKey2000( pScrn ); #endif #if 0 /* Set FIFO L2 on second stream. */ if( pPriv->lastKnownPitch != pitch ) { unsigned char cr92; pPriv->lastKnownPitch = pitch; pitch = (pitch + 7) / 8 - 4; VGAOUT8(vgaCRIndex, 0x92); cr92 = VGAIN8(vgaCRReg); VGAOUT8(vgaCRReg, (cr92 & 0x40) | (pitch >> 8) | 0x80); VGAOUT8(vgaCRIndex, 0x93); VGAOUT8(vgaCRReg, pitch); } #endif } static void SavageFillKeyHelper(DrawablePtr pDraw, uint32_t colorKey, RegionPtr clipBoxes) { #if HAVE_XV_DRAWABLE_HELPER xf86XVFillKeyHelperDrawable(pDraw, colorKey, clipBoxes); #else xf86XVFillKeyHelper(pDraw->pScreen, colorKey, clipBoxes); #endif } static int SavagePutImage( ScrnInfoPtr pScrn, short src_x, short src_y, short drw_x, short drw_y, short src_w, short src_h, short drw_w, short drw_h, int id, unsigned char* buf, short width, short height, Bool sync, RegionPtr clipBoxes, pointer data, DrawablePtr pDraw ){ SavagePortPrivPtr pPriv = (SavagePortPrivPtr)data; SavagePtr psav = SAVPTR(pScrn); #ifdef SAVAGEDRI ScreenPtr pScreen = pScrn->pScreen; #endif INT32 x1, x2, y1, y2; unsigned char *dst_start; int new_size, offset, offsetV=0, offsetU=0; int srcPitch, srcPitch2=0, dstPitch; int planarFrameSize; int top, left, npixels, nlines; BoxRec dstBox; CARD32 tmp; /* xf86ErrorFVerb(XVTRACE,"SavagePutImage\n"); */ if(drw_w > 16384) drw_w = 16384; /* Clip */ x1 = src_x; x2 = src_x + src_w; y1 = src_y; y2 = src_y + src_h; dstBox.x1 = drw_x; dstBox.x2 = drw_x + drw_w; dstBox.y1 = drw_y; dstBox.y2 = drw_y + drw_h; SavageClipVideo(&dstBox, &x1, &x2, &y1, &y2, REGION_EXTENTS(pScreen, clipBoxes), width, height); drw_w = dstBox.x2 - dstBox.x1; drw_h = dstBox.y2 - dstBox.y1; src_w = ( x2 - x1 ) >> 16; src_h = ( y2 - y1 ) >> 16; if((x1 >= x2) || (y1 >= y2)) return Success; dstBox.x1 -= pScrn->frameX0; dstBox.x2 -= pScrn->frameX0; dstBox.y1 -= pScrn->frameY0; dstBox.y2 -= pScrn->frameY0; /* All formats directly displayable by Savage are packed and 2 bytes per pixel */ dstPitch = ((width << 1) + 15) & ~15; new_size = dstPitch * height; switch(id) { case FOURCC_Y211: /* Y211 */ srcPitch = width; break; case FOURCC_YV12: /* YV12 */ srcPitch = (width + 3) & ~3; offsetV = srcPitch * height; srcPitch2 = ((width >> 1) + 3) & ~3; offsetU = (srcPitch2 * (height >> 1)) + offsetV; break; case FOURCC_I420: srcPitch = (width + 3) & ~3; offsetU = srcPitch * height; srcPitch2 = ((width >> 1) + 3) & ~3; offsetV = (srcPitch2 * (height >> 1)) + offsetU; break; case FOURCC_RV15: /* RGB15 */ case FOURCC_RV16: /* RGB16 */ case FOURCC_YUY2: /* YUY2 */ default: srcPitch = (width << 1); break; } /* Calculate required memory for all planar frames */ planarFrameSize = 0; if (srcPitch2 != 0 && S3_SAVAGE4_SERIES(psav->Chipset) && psav->BCIforXv) { new_size = ((new_size + 0xF) & ~0xF); planarFrameSize = srcPitch * height + srcPitch2 * height; } /* Check whether AGP buffers can be allocated. If not, fall back to ordinary upload to framebuffer (slower) */ #ifdef SAVAGEDRI if (!pPriv->tried_agp && !psav->IsPCI && psav->drmFD > 0 && psav->DRIServerInfo != NULL) { SAVAGEDRIServerPrivatePtr pSAVAGEDRIServer = psav->DRIServerInfo; pPriv->tried_agp = TRUE; if (pSAVAGEDRIServer->agpXVideo.size >= max(new_size, planarFrameSize)) { if (pSAVAGEDRIServer->agpXVideo.map == NULL && drmMap( psav->drmFD, pSAVAGEDRIServer->agpXVideo.handle, pSAVAGEDRIServer->agpXVideo.size, &pSAVAGEDRIServer->agpXVideo.map ) < 0 ) { xf86DrvMsg( pScreen->myNum, X_ERROR, "[agp] XVideo: Could not map agpXVideo \n" ); pPriv->agpBufferOffset = 0; pPriv->agpBufferMap = NULL; } else { pPriv->agpBufferMap = pSAVAGEDRIServer->agpXVideo.map; pPriv->agpBufferOffset = pSAVAGEDRIServer->agpXVideo.offset; pPriv->agpBase = drmAgpBase(psav->drmFD); #if 0 xf86DrvMsg( pScreen->myNum, X_INFO, "[agp] agpXVideo mapped at 0x%08lx aperture=0x%08x offset=0x%08lx\n", (unsigned long)pPriv->agpBufferMap, pPriv->agpBase, pPriv->agpBufferOffset); #endif } } else { /* This situation is expected if AGPforXv is disabled, otherwise report. */ if (pSAVAGEDRIServer->agpXVideo.size > 0) { xf86DrvMsg( pScreen->myNum, X_ERROR, "[agp] XVideo: not enough space in buffer (got %ld bytes, required %d bytes).\n", (long int)pSAVAGEDRIServer->agpXVideo.size, max(new_size, planarFrameSize)); } pPriv->agpBufferMap = NULL; pPriv->agpBufferOffset = 0; } } #endif /* SAVAGEDRI */ /* Buffer for final packed frame */ pPriv->video_offset = SavageAllocateMemory( pScrn, &pPriv->video_memory, new_size); if (pPriv->video_offset == 0) return BadAlloc; /* Packed format cases */ if (planarFrameSize == 0) { pPriv->video_planarbuf = 0; /* Planar format cases */ } else { /* Hardware-assisted planar conversion only works on 16-byte aligned addresses */ pPriv->video_planarbuf = SavageAllocateMemory( pScrn, &pPriv->video_planarmem, ((planarFrameSize + 0xF) & ~0xF)); if (pPriv->video_planarbuf != 0) { /* TODO: stop any pending conversions when buffers change... */ pPriv->video_planarbuf = ((pPriv->video_planarbuf + 0xF) & ~0xF); } else { /* Fallback using software conversion */ } } /* copy data */ top = y1 >> 16; left = (x1 >> 16) & ~1; npixels = ((((x2 + 0xffff) >> 16) + 1) & ~1) - left; left <<= 1; offset = (pPriv->video_offset) + (top * dstPitch); dst_start = (psav->FBBase + ((offset + left) & ~BASE_PAD)); switch(id) { case FOURCC_YV12: /* YV12 */ case FOURCC_I420: top &= ~1; tmp = ((top >> 1) * srcPitch2) + (left >> 2); offsetU += tmp; offsetV += tmp; nlines = ((((y2 + 0xffff) >> 16) + 1) & ~1) - top; if (S3_SAVAGE4_SERIES(psav->Chipset) && psav->BCIforXv && (npixels & 0xF) == 0 && pPriv->video_planarbuf != 0) { #ifdef SAVAGEDRI if (pPriv->agpBufferMap != NULL) { /* Using copy to AGP memory */ SavageCopyPlanarDataBCI( pScrn, buf + (top * srcPitch) + (left >> 1), buf + offsetV, buf + offsetU, dst_start, pPriv->agpBufferMap, pPriv->agpBase + pPriv->agpBufferOffset, srcPitch, srcPitch2, dstPitch, nlines, npixels, TRUE); } else #endif /* SAVAGEDRI */ { /* Using ordinary copy to framebuffer */ SavageCopyPlanarDataBCI( pScrn, buf + (top * srcPitch) + (left >> 1), buf + offsetV, buf + offsetU, dst_start, (unsigned char *)psav->FBBase + pPriv->video_planarbuf, pPriv->video_planarbuf, srcPitch, srcPitch2, dstPitch, nlines, npixels, FALSE); } } else { SavageCopyPlanarData( buf + (top * srcPitch) + (left >> 1), buf + offsetV, buf + offsetU, dst_start, srcPitch, srcPitch2, dstPitch, nlines, npixels); } break; case FOURCC_Y211: /* Y211 */ case FOURCC_RV15: /* RGB15 */ case FOURCC_RV16: /* RGB16 */ case FOURCC_YUY2: /* YUY2 */ default: buf += (top * srcPitch) + left; nlines = ((y2 + 0xffff) >> 16) - top; SavageCopyData(buf, dst_start, srcPitch, dstPitch, nlines, npixels); break; } /* We need to enable the video before we draw the chroma color. Otherwise, we get blue flashes. */ SavageDisplayVideo(pScrn, id, offset, width, height, dstPitch, x1, y1, x2, y2, &dstBox, src_w, src_h, drw_w, drw_h); /* update cliplist */ if(!REGION_EQUAL(pScreen, &pPriv->clip, clipBoxes)) { REGION_COPY(pScreen, &pPriv->clip, clipBoxes); /* draw these */ SavageFillKeyHelper(pDraw, pPriv->colorKey, clipBoxes); } pPriv->videoStatus = CLIENT_VIDEO_ON; return Success; } static int SavageQueryImageAttributes( ScrnInfoPtr pScrn, int id, unsigned short *w, unsigned short *h, int *pitches, int *offsets ){ int size, tmp; if(*w > 1024) *w = 1024; if(*h > 1024) *h = 1024; *w = (*w + 1) & ~1; if(offsets) offsets[0] = 0; switch(id) { case FOURCC_IA44: if (pitches) pitches[0]=*w; size=(*w)*(*h); break; case FOURCC_Y211: size = *w << 2; if(pitches) pitches[0] = size; size *= *h; break; case FOURCC_YV12: case FOURCC_I420: *h = (*h + 1) & ~1; size = (*w + 3) & ~3; if(pitches) pitches[0] = size; size *= *h; if(offsets) offsets[1] = size; tmp = ((*w >> 1) + 3) & ~3; if(pitches) pitches[1] = pitches[2] = tmp; tmp *= (*h >> 1); size += tmp; if(offsets) offsets[2] = size; size += tmp; break; case FOURCC_RV15: /* RGB15 */ case FOURCC_RV16: /* RGB16 */ case FOURCC_YUY2: default: size = *w << 1; if(pitches) pitches[0] = size; size *= *h; break; } return size; } /****************** Offscreen stuff ***************/ typedef struct { void *surface_memory; Bool isOn; } OffscreenPrivRec, * OffscreenPrivPtr; static int SavageAllocateSurface( ScrnInfoPtr pScrn, int id, unsigned short w, unsigned short h, XF86SurfacePtr surface ){ int offset, size; int pitch; void *surface_memory = NULL; OffscreenPrivPtr pPriv; if((w > 1024) || (h > 1024)) return BadAlloc; w = (w + 1) & ~1; pitch = ((w << 1) + 15) & ~15; size = pitch * h; offset = SavageAllocateMemory(pScrn, &surface_memory, size); if (offset == 0) return BadAlloc; surface->width = w; surface->height = h; if(!(surface->pitches = malloc(sizeof(int)))) { SavageFreeMemory(pScrn, surface_memory); return BadAlloc; } if(!(surface->offsets = malloc(sizeof(int)))) { free(surface->pitches); SavageFreeMemory(pScrn, surface_memory); return BadAlloc; } if(!(pPriv = malloc(sizeof(OffscreenPrivRec)))) { free(surface->pitches); free(surface->offsets); SavageFreeMemory(pScrn, surface_memory); return BadAlloc; } pPriv->surface_memory = surface_memory; pPriv->isOn = FALSE; surface->pScrn = pScrn; surface->id = id; surface->pitches[0] = pitch; surface->offsets[0] = offset; /*area->box.y1 * fbpitch;*/ surface->devPrivate.ptr = (pointer)pPriv; return Success; } static int SavageStopSurface( XF86SurfacePtr surface ){ OffscreenPrivPtr pPriv = (OffscreenPrivPtr)surface->devPrivate.ptr; xf86ErrorFVerb(XVTRACE,"SavageStopSurface\n"); if(pPriv->isOn) { /*SavagePtr psav = SAVPTR(surface->pScrn);*/ /*SavageClipVWindow(surface->pScrn);*/ SavageStreamsOff( surface->pScrn ); pPriv->isOn = FALSE; } return Success; } static int SavageFreeSurface( XF86SurfacePtr surface ){ ScrnInfoPtr pScrn = surface->pScrn; OffscreenPrivPtr pPriv = (OffscreenPrivPtr)surface->devPrivate.ptr; if(pPriv->isOn) SavageStopSurface(surface); SavageFreeMemory(pScrn, pPriv->surface_memory); free(surface->pitches); free(surface->offsets); free(surface->devPrivate.ptr); return Success; } static int SavageGetSurfaceAttribute( ScrnInfoPtr pScrn, Atom attribute, INT32 *value ){ return SavageGetPortAttribute(pScrn, attribute, value, (pointer)(GET_PORT_PRIVATE(pScrn))); } static int SavageSetSurfaceAttribute( ScrnInfoPtr pScrn, Atom attribute, INT32 value ){ return SavageSetPortAttribute(pScrn, attribute, value, (pointer)(GET_PORT_PRIVATE(pScrn))); } static int SavageDisplaySurface( XF86SurfacePtr surface, short src_x, short src_y, short drw_x, short drw_y, short src_w, short src_h, short drw_w, short drw_h, RegionPtr clipBoxes ){ OffscreenPrivPtr pPriv = (OffscreenPrivPtr)surface->devPrivate.ptr; ScrnInfoPtr pScrn = surface->pScrn; ScreenPtr pScreen = pScrn->pScreen; SavagePortPrivPtr portPriv = GET_PORT_PRIVATE(pScrn); INT32 x1, y1, x2, y2; BoxRec dstBox; xf86ErrorFVerb(XVTRACE,"SavageDisplaySurface\n"); x1 = src_x; x2 = src_x + src_w; y1 = src_y; y2 = src_y + src_h; dstBox.x1 = drw_x; dstBox.x2 = drw_x + drw_w; dstBox.y1 = drw_y; dstBox.y2 = drw_y + drw_h; SavageClipVideo(&dstBox, &x1, &x2, &y1, &y2, REGION_EXTENTS(pScreen, clipBoxes), surface->width, surface->height); if((x1 >= x2) || (y1 >= y2)) return Success; dstBox.x1 -= pScrn->frameX0; dstBox.x2 -= pScrn->frameX0; dstBox.y1 -= pScrn->frameY0; dstBox.y2 -= pScrn->frameY0; SavageDisplayVideo(pScrn, surface->id, surface->offsets[0], surface->width, surface->height, surface->pitches[0], x1, y1, x2, y2, &dstBox, src_w, src_h, drw_w, drw_h); xf86XVFillKeyHelper(pScreen, portPriv->colorKey, clipBoxes); pPriv->isOn = TRUE; #if 0 if(portPriv->videoStatus & CLIENT_VIDEO_ON) { REGION_EMPTY(pScreen, &portPriv->clip); UpdateCurrentTime(); portPriv->videoStatus = FREE_TIMER; portPriv->freeTime = currentTime.milliseconds + FREE_DELAY; } #endif return Success; } static void SavageInitOffscreenImages(ScreenPtr pScreen) { XF86OffscreenImagePtr offscreenImages; SavagePtr psav = SAVPTR(xf86ScreenToScrn(pScreen)); /* need to free this someplace */ if (!psav->offscreenImages) { if(!(offscreenImages = malloc(sizeof(XF86OffscreenImageRec)))) return; psav->offscreenImages = offscreenImages; } else { offscreenImages = psav->offscreenImages; } offscreenImages[0].image = &Images[0]; offscreenImages[0].flags = VIDEO_OVERLAID_IMAGES | VIDEO_CLIP_TO_VIEWPORT; offscreenImages[0].alloc_surface = SavageAllocateSurface; offscreenImages[0].free_surface = SavageFreeSurface; offscreenImages[0].display = SavageDisplaySurface; offscreenImages[0].stop = SavageStopSurface; offscreenImages[0].setAttribute = SavageSetSurfaceAttribute; offscreenImages[0].getAttribute = SavageGetSurfaceAttribute; offscreenImages[0].max_width = 1024; offscreenImages[0].max_height = 1024; offscreenImages[0].num_attributes = NUM_ATTRIBUTES; offscreenImages[0].attributes = Attributes; xf86XVRegisterOffscreenImages(pScreen, offscreenImages, 1); }