2016-12-15 96 views
0

我嘗試使用renderscript中的ScriptIntrinsicYuvToRGB類將YUV轉換爲RGB,其中源爲YUV420格式。 我有3個原始平面,我從文件中讀取並嘗試將它們饋送到YUV類型的Allocation中,並通過ScriptIntrinsicYuvToRGB.forEach傳遞它。ScriptIntrinsicYuvToRGB或YUV420分配是否被破壞?

它正確地轉換亮度(Y平面),但未能上的顏色,因爲色度通道似乎讀取的所有值從BUF [W * H]位置 - 見代碼示例中註釋部分。它看起來像分配不正確地處理UV平面時的錯誤。我假設是因爲我在一個使用rsGetElementAtYuv_uchar_U函數的腳本中進行了測試,並且它對任何座標給出了相同的值(來自buf [w * h])。

我搜索了所有的地方,如果我可以進一步指定YUV格式,如步幅/偏移等,但沒有發現更多的設置Element.DataKind.PIXEL_YUVType.Builder.setYuvFormat(ImageFormat.YUV_420_888)

有人可以幫忙嗎?

{ 
     int w = 320, h = 172; 
     ScriptIntrinsicYuvToRGB yc = ScriptIntrinsicYuvToRGB.create(rs, Element.U8_4(rs)); 
     { 
     Element elemYUV = Element.createPixel(rs, Element.DataType.UNSIGNED_8, Element.DataKind.PIXEL_YUV); 
     Type typeYUV = new Type.Builder(rs, elemYUV).setX(w).setY(h).setYuvFormat(ImageFormat.YUV_420_888).create(); 
     Allocation yuv = Allocation.createTyped(rs, typeYUV); 
     byte[] buf = new byte[yuv.getBytesSize()]; 
     int offs = 0; 
     for(int i=0; i<3; i++){ 
      int sz = w*h; 
      if(i>0) 
       sz /= 4; 
      InputStream is = new FileInputStream("/sdcard/yuv/"+(i==0 ? 'y' : i==1 ? 'u' : 'v')); 
      int n = is.read(buf, offs, sz); 
      if(n!=sz) 
       throw new AssertionError("!"); 
      offs += sz; 
      is.close(); 
     } 
//    buf[w*h] = 0x40; 
     yuv.copyFrom(buf); 
     yc.setInput(yuv); 
     } 

     Type outType = new Type.Builder(rs, Element.U8_4(rs)).setX(w).setY(h).create(); 
     Allocation out = Allocation.createTyped(rs, outType); 
     yc.forEach(out); 

     int[] buf = new int[out.getBytesSize()/4]; 
     out.copy1DRangeToUnchecked(0, w*h, buf); 

     bm = Bitmap.createBitmap(w, h, Bitmap.Config.ARGB_8888); 
     bm.setPixels(buf, 0, bm.getWidth(), 0, 0, bm.getWidth(), bm.getHeight()); 
     iv.setImageBitmap(bm); 

     yc.destroy(); 
    } 

回答

0

一種解決方法是隻需填寫一個U8分配和做自己的索引中的自定義腳本:

#pragma rs_fp_relaxed 
rs_allocation yuv_in; 
uint32_t width; 
uint32_t offset_to_u; 
uint32_t offset_to_v; 
uchar4 RS_KERNEL yuv_to_rgba(uint32_t x, uint32_t y) { 
    uint32_t index = y * width + x; 
    uint32_t uv_index = (y >> 1) * width + (x >> 1); 
    float Y = (float)rsGetElementAt_uchar(yuv_in, index); 
    float U = (float)rsGetElementAt_uchar(yuv_in, uv_index + offset_to_u); 
    float V = (float)rsGetElementAt_uchar(yuv_in, uv_index + offset_to_v); 
    float3 f_out; 
    f_out.r = Y + 1.403f * V; 
    f_out.g = Y - 0.344f * U - 0.714f * V; 
    f_out.b = Y + 1.770f * U; 
    f_out = clamp(f_out, 0.f, 255.f); 
    uchar4 out; 
    out.rgb = convert_uchar3(f_out); 
    out.a = 255; 
    return out; 
} 

的java:

sc.set_yuv_in(yuv_allocation); 
sc.set_width(width); 
sc.set_offset_to_u(width * height); 
sc.set_offset_to_v(width * height + (width/2 * height/2)); 
sc.forEach_yuv_to_rba(out); 

YUV_420_888更多的是一種通用的YUV鍵入從android中的其他YUV資源導入。我認爲沒有辦法設置u/v平面的步距/偏移值,以使其對自定義轉換有用。