Naseer Ahmed | 29a2681 | 2012-06-14 00:56:20 -0700 | [diff] [blame] | 1 | /* |
Praveena Pachipulusu | aef031c | 2014-02-12 11:46:39 +0530 | [diff] [blame] | 2 | * Copyright (c) 2011-2014, The Linux Foundation. All rights reserved. |
Naseer Ahmed | 29a2681 | 2012-06-14 00:56:20 -0700 | [diff] [blame] | 3 | * |
| 4 | * Redistribution and use in source and binary forms, with or without |
| 5 | * modification, are permitted provided that the following conditions are |
| 6 | * met: |
| 7 | * * Redistributions of source code must retain the above copyright |
| 8 | * notice, this list of conditions and the following disclaimer. |
| 9 | * * Redistributions in binary form must reproduce the above |
| 10 | * copyright notice, this list of conditions and the following |
| 11 | * disclaimer in the documentation and/or other materials provided |
| 12 | * with the distribution. |
Duy Truong | 73d36df | 2013-02-09 20:33:23 -0800 | [diff] [blame] | 13 | * * Neither the name of The Linux Foundation nor the names of its |
Naseer Ahmed | 29a2681 | 2012-06-14 00:56:20 -0700 | [diff] [blame] | 14 | * contributors may be used to endorse or promote products derived |
| 15 | * from this software without specific prior written permission. |
| 16 | * |
| 17 | * THIS SOFTWARE IS PROVIDED "AS IS" AND ANY EXPRESS OR IMPLIED |
| 18 | * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF |
| 19 | * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT |
| 20 | * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS |
| 21 | * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR |
| 22 | * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF |
| 23 | * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR |
| 24 | * BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, |
| 25 | * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE |
| 26 | * OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN |
| 27 | * IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
| 28 | */ |
| 29 | |
Naseer Ahmed | 29a2681 | 2012-06-14 00:56:20 -0700 | [diff] [blame] | 30 | #include <cutils/log.h> |
| 31 | #include <stdlib.h> |
| 32 | #include <errno.h> |
| 33 | #include "software_converter.h" |
| 34 | |
| 35 | /** Convert YV12 to YCrCb_420_SP */ |
| 36 | int convertYV12toYCrCb420SP(const copybit_image_t *src, private_handle_t *yv12_handle) |
| 37 | { |
| 38 | private_handle_t* hnd = (private_handle_t*)src->handle; |
| 39 | |
| 40 | if(hnd == NULL || yv12_handle == NULL){ |
| 41 | ALOGE("Invalid handle"); |
| 42 | return -1; |
| 43 | } |
| 44 | |
| 45 | // Please refer to the description of YV12 in hardware.h |
| 46 | // for the formulae used to calculate buffer sizes and offsets |
| 47 | |
| 48 | // In a copybit_image_t, w is the stride and |
| 49 | // stride - horiz_padding is the actual width |
| 50 | // vertical stride is the same as height, so not considered |
| 51 | unsigned int stride = src->w; |
| 52 | unsigned int width = src->w - src->horiz_padding; |
| 53 | unsigned int height = src->h; |
| 54 | unsigned int y_size = stride * src->h; |
Praveena Pachipulusu | aef031c | 2014-02-12 11:46:39 +0530 | [diff] [blame] | 55 | unsigned int c_width = ALIGN(stride/2, (unsigned int)16); |
Naseer Ahmed | 29a2681 | 2012-06-14 00:56:20 -0700 | [diff] [blame] | 56 | unsigned int c_size = c_width * src->h/2; |
| 57 | unsigned int chromaPadding = c_width - width/2; |
| 58 | unsigned int chromaSize = c_size * 2; |
| 59 | unsigned char* newChroma = (unsigned char *)(yv12_handle->base + y_size); |
| 60 | unsigned char* oldChroma = (unsigned char*)(hnd->base + y_size); |
| 61 | memcpy((char *)yv12_handle->base,(char *)hnd->base,y_size); |
| 62 | |
Sushil Chauhan | 5cc90dd | 2014-07-31 12:22:31 -0700 | [diff] [blame] | 63 | #if defined(__ARM_HAVE_NEON) && !defined(__aarch64__) |
Naseer Ahmed | 29a2681 | 2012-06-14 00:56:20 -0700 | [diff] [blame] | 64 | /* interleave */ |
| 65 | if(!chromaPadding) { |
| 66 | unsigned char * t1 = newChroma; |
| 67 | unsigned char * t2 = oldChroma; |
| 68 | unsigned char * t3 = t2 + chromaSize/2; |
| 69 | for(unsigned int i=0; i < (chromaSize/2)>>3; i++) { |
| 70 | __asm__ __volatile__ ( |
| 71 | "vld1.u8 d0, [%0]! \n" |
| 72 | "vld1.u8 d1, [%1]! \n" |
| 73 | "vst2.u8 {d0, d1}, [%2]! \n" |
| 74 | :"+r"(t2), "+r"(t3), "+r"(t1) |
| 75 | : |
| 76 | :"memory","d0","d1" |
| 77 | ); |
| 78 | |
| 79 | } |
| 80 | } |
| 81 | #else //__ARM_HAVE_NEON |
| 82 | if(!chromaPadding) { |
| 83 | for(unsigned int i = 0; i< chromaSize/2; i++) { |
| 84 | newChroma[i*2] = oldChroma[i]; |
| 85 | newChroma[i*2+1] = oldChroma[i+chromaSize/2]; |
| 86 | } |
| 87 | |
| 88 | } |
| 89 | #endif |
| 90 | // If the image is not aligned to 16 pixels, |
| 91 | // convert using the C routine below |
| 92 | // r1 tracks the row of the source buffer |
| 93 | // r2 tracks the row of the destination buffer |
| 94 | // The width/2 checks are to avoid copying |
| 95 | // from the padding |
| 96 | |
| 97 | if(chromaPadding) { |
| 98 | unsigned int r1 = 0, r2 = 0, i = 0, j = 0; |
| 99 | while(r1 < height/2) { |
| 100 | if(j == width) { |
| 101 | j = 0; |
| 102 | r2++; |
| 103 | continue; |
| 104 | } |
| 105 | if (j+1 == width) { |
| 106 | newChroma[r2*width + j] = oldChroma[r1*c_width+i]; |
| 107 | r2++; |
| 108 | newChroma[r2*width] = oldChroma[r1*c_width+i+c_size]; |
| 109 | j = 1; |
| 110 | } else { |
| 111 | newChroma[r2*width + j] = oldChroma[r1*c_width+i]; |
| 112 | newChroma[r2*width + j + 1] = oldChroma[r1*c_width+i+c_size]; |
| 113 | j+=2; |
| 114 | } |
| 115 | i++; |
| 116 | if (i == width/2 ) { |
| 117 | i = 0; |
| 118 | r1++; |
| 119 | } |
| 120 | } |
| 121 | } |
| 122 | |
| 123 | return 0; |
| 124 | } |
| 125 | |
| 126 | struct copyInfo{ |
| 127 | int width; |
| 128 | int height; |
| 129 | int src_stride; |
| 130 | int dst_stride; |
Praveena Pachipulusu | aef031c | 2014-02-12 11:46:39 +0530 | [diff] [blame] | 131 | size_t src_plane1_offset; |
| 132 | size_t src_plane2_offset; |
| 133 | size_t dst_plane1_offset; |
| 134 | size_t dst_plane2_offset; |
Naseer Ahmed | 29a2681 | 2012-06-14 00:56:20 -0700 | [diff] [blame] | 135 | }; |
| 136 | |
| 137 | /* Internal function to do the actual copy of source to destination */ |
Praveena Pachipulusu | aef031c | 2014-02-12 11:46:39 +0530 | [diff] [blame] | 138 | static int copy_source_to_destination(const uintptr_t src_base, |
| 139 | const uintptr_t dst_base, |
Naseer Ahmed | 29a2681 | 2012-06-14 00:56:20 -0700 | [diff] [blame] | 140 | copyInfo& info) |
| 141 | { |
| 142 | if (!src_base || !dst_base) { |
Praveena Pachipulusu | aef031c | 2014-02-12 11:46:39 +0530 | [diff] [blame] | 143 | ALOGE("%s: invalid memory src_base = 0x%p dst_base=0x%p", |
| 144 | __FUNCTION__, (void*)src_base, (void*)dst_base); |
Naseer Ahmed | 29a2681 | 2012-06-14 00:56:20 -0700 | [diff] [blame] | 145 | return COPYBIT_FAILURE; |
| 146 | } |
| 147 | |
| 148 | int width = info.width; |
| 149 | int height = info.height; |
| 150 | unsigned char *src = (unsigned char*)src_base; |
| 151 | unsigned char *dst = (unsigned char*)dst_base; |
| 152 | |
| 153 | // Copy the luma |
| 154 | for (int i = 0; i < height; i++) { |
| 155 | memcpy(dst, src, width); |
| 156 | src += info.src_stride; |
| 157 | dst += info.dst_stride; |
| 158 | } |
| 159 | |
| 160 | // Copy plane 1 |
| 161 | src = (unsigned char*)(src_base + info.src_plane1_offset); |
| 162 | dst = (unsigned char*)(dst_base + info.dst_plane1_offset); |
| 163 | width = width/2; |
| 164 | height = height/2; |
| 165 | for (int i = 0; i < height; i++) { |
| 166 | memcpy(dst, src, info.src_stride); |
| 167 | src += info.src_stride; |
| 168 | dst += info.dst_stride; |
| 169 | } |
| 170 | return 0; |
| 171 | } |
| 172 | |
| 173 | |
| 174 | /* |
| 175 | * Function to convert the c2d format into an equivalent Android format |
| 176 | * |
| 177 | * @param: source buffer handle |
| 178 | * @param: destination image |
| 179 | * |
| 180 | * @return: return status |
| 181 | */ |
| 182 | int convert_yuv_c2d_to_yuv_android(private_handle_t *hnd, |
| 183 | struct copybit_image_t const *rhs) |
| 184 | { |
| 185 | ALOGD("Enter %s", __FUNCTION__); |
| 186 | if (!hnd || !rhs) { |
| 187 | ALOGE("%s: invalid inputs hnd=%p rhs=%p", __FUNCTION__, hnd, rhs); |
| 188 | return COPYBIT_FAILURE; |
| 189 | } |
| 190 | |
| 191 | int ret = COPYBIT_SUCCESS; |
| 192 | private_handle_t *dst_hnd = (private_handle_t *)rhs->handle; |
| 193 | |
| 194 | copyInfo info; |
| 195 | info.width = rhs->w; |
| 196 | info.height = rhs->h; |
| 197 | info.src_stride = ALIGN(info.width, 32); |
| 198 | info.dst_stride = ALIGN(info.width, 16); |
| 199 | switch(rhs->format) { |
| 200 | case HAL_PIXEL_FORMAT_YCbCr_420_SP: |
| 201 | case HAL_PIXEL_FORMAT_YCrCb_420_SP: { |
| 202 | info.src_plane1_offset = info.src_stride*info.height; |
| 203 | info.dst_plane1_offset = info.dst_stride*info.height; |
| 204 | } break; |
| 205 | case HAL_PIXEL_FORMAT_NV12_ENCODEABLE: { |
| 206 | // Chroma is 2K aligned for the NV12 encodeable format. |
| 207 | info.src_plane1_offset = ALIGN(info.src_stride*info.height, 2048); |
| 208 | info.dst_plane1_offset = ALIGN(info.dst_stride*info.height, 2048); |
| 209 | } break; |
| 210 | default: |
| 211 | ALOGE("%s: unsupported format (format=0x%x)", __FUNCTION__, |
| 212 | rhs->format); |
| 213 | return COPYBIT_FAILURE; |
| 214 | } |
| 215 | |
Naseer Ahmed | 9eb5e09 | 2014-09-25 13:24:44 -0400 | [diff] [blame] | 216 | ret = copy_source_to_destination((uintptr_t) hnd->base, (uintptr_t) dst_hnd->base, info); |
Naseer Ahmed | 29a2681 | 2012-06-14 00:56:20 -0700 | [diff] [blame] | 217 | return ret; |
| 218 | } |
| 219 | |
| 220 | /* |
| 221 | * Function to convert the Android format into an equivalent C2D format |
| 222 | * |
| 223 | * @param: source buffer handle |
| 224 | * @param: destination image |
| 225 | * |
| 226 | * @return: return status |
| 227 | */ |
| 228 | int convert_yuv_android_to_yuv_c2d(private_handle_t *hnd, |
| 229 | struct copybit_image_t const *rhs) |
| 230 | { |
| 231 | if (!hnd || !rhs) { |
| 232 | ALOGE("%s: invalid inputs hnd=%p rhs=%p", __FUNCTION__, hnd, rhs); |
| 233 | return COPYBIT_FAILURE; |
| 234 | } |
| 235 | |
| 236 | int ret = COPYBIT_SUCCESS; |
| 237 | private_handle_t *dst_hnd = (private_handle_t *)rhs->handle; |
| 238 | |
| 239 | copyInfo info; |
| 240 | info.width = rhs->w; |
| 241 | info.height = rhs->h; |
| 242 | info.src_stride = ALIGN(hnd->width, 16); |
| 243 | info.dst_stride = ALIGN(info.width, 32); |
| 244 | switch(rhs->format) { |
| 245 | case HAL_PIXEL_FORMAT_YCbCr_420_SP: |
| 246 | case HAL_PIXEL_FORMAT_YCrCb_420_SP: { |
| 247 | info.src_plane1_offset = info.src_stride*info.height; |
| 248 | info.dst_plane1_offset = info.dst_stride*info.height; |
| 249 | } break; |
| 250 | case HAL_PIXEL_FORMAT_NV12_ENCODEABLE: { |
| 251 | // Chroma is 2K aligned for the NV12 encodeable format. |
| 252 | info.src_plane1_offset = ALIGN(info.src_stride*info.height, 2048); |
| 253 | info.dst_plane1_offset = ALIGN(info.dst_stride*info.height, 2048); |
| 254 | } break; |
| 255 | default: |
| 256 | ALOGE("%s: unsupported format (format=0x%x)", __FUNCTION__, |
| 257 | rhs->format); |
| 258 | return -1; |
| 259 | } |
| 260 | |
Naseer Ahmed | 9eb5e09 | 2014-09-25 13:24:44 -0400 | [diff] [blame] | 261 | ret = copy_source_to_destination((uintptr_t) hnd->base, (uintptr_t) dst_hnd->base, info); |
Naseer Ahmed | 29a2681 | 2012-06-14 00:56:20 -0700 | [diff] [blame] | 262 | return ret; |
| 263 | } |