显示RGB888内容

时间:2010-03-12 10:35:13

标签: c++ c windows-mobile rgb windows-ce

我必须使用 ShowRGBContent 功能显示 RGB888 内容。

以下功能是 yv12-> rgb565 &的 ShowRGBContent 功能。的 UYVY-> RGB565

static void ShowRGBContent(UINT8 * pImageBuf, INT32 width, INT32 height) 
{
LogEntry(L"%d : In %s Function \r\n",++abhineet,__WFUNCTION__);
UINT16 * temp;
BYTE rValue, gValue, bValue;

// this is to refresh the background desktop
ShowWindow(GetDesktopWindow(),SW_HIDE);
ShowWindow(GetDesktopWindow(),SW_SHOW); 

for(int i=0; i<height; i++)
{
 for (int j=0; j< width; j++)        
 {
  temp = (UINT16 *) (pImageBuf+ i*width*PP_TEST_FRAME_BPP+j*PP_TEST_FRAME_BPP);
  bValue = (BYTE) ((*temp & RGB_COMPONET0_MASK) >> RGB_COMPONET0_OFFSET) << (8 -RGB_COMPONET0_WIDTH);
  gValue = (BYTE) ((*temp & RGB_COMPONET1_MASK) >> RGB_COMPONET1_OFFSET) << (8 -RGB_COMPONET1_WIDTH);
  rValue = (BYTE) ((*temp & RGB_COMPONET2_MASK) >> RGB_COMPONET2_OFFSET) << (8 -RGB_COMPONET2_WIDTH);            
  SetPixel(g_hDisplay, SCREEN_OFFSET_X + j, SCREEN_OFFSET_Y+i, RGB(rValue, gValue, bValue));
 }
}

Sleep(2000); //sleep here to review the result

LogEntry(L"%d :Out %s Function \r\n",++abhineet,__WFUNCTION__);
}

我必须为RGB888修改此内容

以上功能:

************************
RGB_COMPONET0_WIDTH = 5
RGB_COMPONET1_WIDTH = 6
RGB_COMPONET2_WIDTH = 5
************************

************************
RGB_COMPONET0_MASK = 0x001F //31 in decimal
RGB_COMPONET1_MASK = 0x07E0 //2016 in decimal
RGB_COMPONET2_MASK = 0xF800 //63488 in decimal
************************

************************
RGB_COMPONET0_OFFSET = 0
RGB_COMPONET1_OFFSET = 5
RGB_COMPONET2_OFFSET = 11
************************

************************
SCREEN_OFFSET_X = 100
SCREEN_OFFSET_Y = 0
************************

Here 
Also PP_TEST_FRAME_BPP = 2 for yv12 -> RGB565 & UYVY -> RGB565

iOutputBytesPerFrame = iOutputStride * iOutputHeight; 
// where iOutputStride = (iOutputWidth * PP_TEST_FRAME_BPP) i.e (112 * 2)  
// & iOutputHeight = 160 
// These are in case of RGB565

pOutputFrameVirtAddr = (UINT32 *) AllocPhysMem( iOutputBytesPerFrame, 
                                                PAGE_EXECUTE_READWRITE, 
                                                0,
                                                0,
                                                (ULONG *) &pOutputFramePhysAddr); 

// PAGE_EXECUTE_READWRITE = 0x40 mentioned in winnt.h

// Width =112 & Height = 160 in all the formats for i/p & o/p

现在我的任务是RGB888。 请指导我该怎么做。 **提前致谢。

1 个答案:

答案 0 :(得分:1)

从yuv444到rgb888的转换非常简单,因为所有组件都落在字节边界上,因此甚至不需要进行位屏蔽。根据评论部分中提到的wikipedia article nobugz,转换可以通过以下方式在固定点完成

UINT8* pimg = pImageBuf;
for(int i=0; i<height; i++)
{
    for (int j=0; j< width; j++)        
    {
        INT16 Y = pimg[0];
        INT16 Cb = (INT16)pimg[1] - 128;
        INT16 Cr = (INT16)pimg[2] - 128;
        rValue = Y + Cr + Cr >> 2 + Cr >> 3 + Cr >> 5
        gValue = Y - (Cb >> 2 + Cb >> 4 + Cb >> 5) -
                 (Cr >> 1 + Cr >> 3 + Cr >> 4 + Cr >> 5);
        bValue = Y + Cb + Cb >> 1 + Cb >> 2 + Cb >> 6;
        SetPixel(g_hDisplay, SCREEN_OFFSET_X + j, SCREEN_OFFSET_Y+i, RGB(rValue,
                 gValue, bValue));
        pimg+=3;
    }
}

这假设你的yuv444是每个样本8位(每像素24位)。转换也可以在浮点运行,但如果它可以工作,则应该更快,因为源和目标都是固定点。我也不确定转换到int16是否必要,但我这样做是为了安全。

请注意,yuv444中的444与rgb888中的888不同。 444指的是使用TUV颜色空间时经常出现的子采样。例如,在YUV420中,Cb和Cr在两个方向上被两个抽取。 yuv444只意味着所有三个组件都采样相同(没有子采样)。 rgb888中的888指的是每个样本的位数(三个颜色分量中的每一个都是8位)。

我实际上没有测试过这段代码,但它至少应该让你知道从哪里开始。