media/libyuv/source/convert_argb.cc

Thu, 22 Jan 2015 13:21:57 +0100

author
Michael Schloh von Bennewitz <michael@schloh.com>
date
Thu, 22 Jan 2015 13:21:57 +0100
branch
TOR_BUG_9701
changeset 15
b8a032363ba2
permissions
-rw-r--r--

Incorporate requested changes from Mozilla in review:
https://bugzilla.mozilla.org/show_bug.cgi?id=1123480#c6

     1 /*
     2  *  Copyright 2011 The LibYuv Project Authors. All rights reserved.
     3  *
     4  *  Use of this source code is governed by a BSD-style license
     5  *  that can be found in the LICENSE file in the root of the source
     6  *  tree. An additional intellectual property rights grant can be found
     7  *  in the file PATENTS. All contributing project authors may
     8  *  be found in the AUTHORS file in the root of the source tree.
     9  */
    11 #include "libyuv/convert_argb.h"
    13 #include "libyuv/cpu_id.h"
    14 #include "libyuv/format_conversion.h"
    15 #ifdef HAVE_JPEG
    16 #include "libyuv/mjpeg_decoder.h"
    17 #endif
    18 #include "libyuv/rotate_argb.h"
    19 #include "libyuv/row.h"
    20 #include "libyuv/video_common.h"
    22 #ifdef __cplusplus
    23 namespace libyuv {
    24 extern "C" {
    25 #endif
    27 // Copy ARGB with optional flipping
    28 LIBYUV_API
    29 int ARGBCopy(const uint8* src_argb, int src_stride_argb,
    30              uint8* dst_argb, int dst_stride_argb,
    31              int width, int height) {
    32   if (!src_argb || !dst_argb ||
    33       width <= 0 || height == 0) {
    34     return -1;
    35   }
    36   // Negative height means invert the image.
    37   if (height < 0) {
    38     height = -height;
    39     src_argb = src_argb + (height - 1) * src_stride_argb;
    40     src_stride_argb = -src_stride_argb;
    41   }
    43   CopyPlane(src_argb, src_stride_argb, dst_argb, dst_stride_argb,
    44             width * 4, height);
    45   return 0;
    46 }
    48 // Convert I444 to ARGB.
    49 LIBYUV_API
    50 int I444ToARGB(const uint8* src_y, int src_stride_y,
    51                const uint8* src_u, int src_stride_u,
    52                const uint8* src_v, int src_stride_v,
    53                uint8* dst_argb, int dst_stride_argb,
    54                int width, int height) {
    55   if (!src_y || !src_u || !src_v ||
    56       !dst_argb ||
    57       width <= 0 || height == 0) {
    58     return -1;
    59   }
    60   // Negative height means invert the image.
    61   if (height < 0) {
    62     height = -height;
    63     dst_argb = dst_argb + (height - 1) * dst_stride_argb;
    64     dst_stride_argb = -dst_stride_argb;
    65   }
    66   // Coalesce rows.
    67   if (src_stride_y == width &&
    68       src_stride_u == width &&
    69       src_stride_v == width &&
    70       dst_stride_argb == width * 4) {
    71     width *= height;
    72     height = 1;
    73     src_stride_y = src_stride_u = src_stride_v = dst_stride_argb = 0;
    74   }
    75   void (*I444ToARGBRow)(const uint8* y_buf,
    76                         const uint8* u_buf,
    77                         const uint8* v_buf,
    78                         uint8* rgb_buf,
    79                         int width) = I444ToARGBRow_C;
    80 #if defined(HAS_I444TOARGBROW_SSSE3)
    81   if (TestCpuFlag(kCpuHasSSSE3) && width >= 8) {
    82     I444ToARGBRow = I444ToARGBRow_Any_SSSE3;
    83     if (IS_ALIGNED(width, 8)) {
    84       I444ToARGBRow = I444ToARGBRow_Unaligned_SSSE3;
    85       if (IS_ALIGNED(dst_argb, 16) && IS_ALIGNED(dst_stride_argb, 16)) {
    86         I444ToARGBRow = I444ToARGBRow_SSSE3;
    87       }
    88     }
    89   }
    90 #elif defined(HAS_I444TOARGBROW_NEON)
    91   if (TestCpuFlag(kCpuHasNEON) && width >= 8) {
    92     I444ToARGBRow = I444ToARGBRow_Any_NEON;
    93     if (IS_ALIGNED(width, 8)) {
    94       I444ToARGBRow = I444ToARGBRow_NEON;
    95     }
    96   }
    97 #endif
    99   for (int y = 0; y < height; ++y) {
   100     I444ToARGBRow(src_y, src_u, src_v, dst_argb, width);
   101     dst_argb += dst_stride_argb;
   102     src_y += src_stride_y;
   103     src_u += src_stride_u;
   104     src_v += src_stride_v;
   105   }
   106   return 0;
   107 }
   109 // Convert I422 to ARGB.
   110 LIBYUV_API
   111 int I422ToARGB(const uint8* src_y, int src_stride_y,
   112                const uint8* src_u, int src_stride_u,
   113                const uint8* src_v, int src_stride_v,
   114                uint8* dst_argb, int dst_stride_argb,
   115                int width, int height) {
   116   if (!src_y || !src_u || !src_v ||
   117       !dst_argb ||
   118       width <= 0 || height == 0) {
   119     return -1;
   120   }
   121   // Negative height means invert the image.
   122   if (height < 0) {
   123     height = -height;
   124     dst_argb = dst_argb + (height - 1) * dst_stride_argb;
   125     dst_stride_argb = -dst_stride_argb;
   126   }
   127   // Coalesce rows.
   128   if (src_stride_y == width &&
   129       src_stride_u * 2 == width &&
   130       src_stride_v * 2 == width &&
   131       dst_stride_argb == width * 4) {
   132     width *= height;
   133     height = 1;
   134     src_stride_y = src_stride_u = src_stride_v = dst_stride_argb = 0;
   135   }
   136   void (*I422ToARGBRow)(const uint8* y_buf,
   137                         const uint8* u_buf,
   138                         const uint8* v_buf,
   139                         uint8* rgb_buf,
   140                         int width) = I422ToARGBRow_C;
   141 #if defined(HAS_I422TOARGBROW_SSSE3)
   142   if (TestCpuFlag(kCpuHasSSSE3) && width >= 8) {
   143     I422ToARGBRow = I422ToARGBRow_Any_SSSE3;
   144     if (IS_ALIGNED(width, 8)) {
   145       I422ToARGBRow = I422ToARGBRow_Unaligned_SSSE3;
   146       if (IS_ALIGNED(dst_argb, 16) && IS_ALIGNED(dst_stride_argb, 16)) {
   147         I422ToARGBRow = I422ToARGBRow_SSSE3;
   148       }
   149     }
   150   }
   151 #endif
   152 #if defined(HAS_I422TOARGBROW_AVX2)
   153   if (TestCpuFlag(kCpuHasAVX2) && width >= 16) {
   154     I422ToARGBRow = I422ToARGBRow_Any_AVX2;
   155     if (IS_ALIGNED(width, 16)) {
   156       I422ToARGBRow = I422ToARGBRow_AVX2;
   157     }
   158   }
   159 #endif
   160 #if defined(HAS_I422TOARGBROW_NEON)
   161   if (TestCpuFlag(kCpuHasNEON) && width >= 8) {
   162     I422ToARGBRow = I422ToARGBRow_Any_NEON;
   163     if (IS_ALIGNED(width, 8)) {
   164       I422ToARGBRow = I422ToARGBRow_NEON;
   165     }
   166   }
   167 #endif
   168 #if defined(HAS_I422TOARGBROW_MIPS_DSPR2)
   169   if (TestCpuFlag(kCpuHasMIPS_DSPR2) && IS_ALIGNED(width, 4) &&
   170       IS_ALIGNED(src_y, 4) && IS_ALIGNED(src_stride_y, 4) &&
   171       IS_ALIGNED(src_u, 2) && IS_ALIGNED(src_stride_u, 2) &&
   172       IS_ALIGNED(src_v, 2) && IS_ALIGNED(src_stride_v, 2) &&
   173       IS_ALIGNED(dst_argb, 4) && IS_ALIGNED(dst_stride_argb, 4)) {
   174     I422ToARGBRow = I422ToARGBRow_MIPS_DSPR2;
   175   }
   176 #endif
   178   for (int y = 0; y < height; ++y) {
   179     I422ToARGBRow(src_y, src_u, src_v, dst_argb, width);
   180     dst_argb += dst_stride_argb;
   181     src_y += src_stride_y;
   182     src_u += src_stride_u;
   183     src_v += src_stride_v;
   184   }
   185   return 0;
   186 }
   188 // Convert I411 to ARGB.
   189 LIBYUV_API
   190 int I411ToARGB(const uint8* src_y, int src_stride_y,
   191                const uint8* src_u, int src_stride_u,
   192                const uint8* src_v, int src_stride_v,
   193                uint8* dst_argb, int dst_stride_argb,
   194                int width, int height) {
   195   if (!src_y || !src_u || !src_v ||
   196       !dst_argb ||
   197       width <= 0 || height == 0) {
   198     return -1;
   199   }
   200   // Negative height means invert the image.
   201   if (height < 0) {
   202     height = -height;
   203     dst_argb = dst_argb + (height - 1) * dst_stride_argb;
   204     dst_stride_argb = -dst_stride_argb;
   205   }
   206   // Coalesce rows.
   207   if (src_stride_y == width &&
   208       src_stride_u * 4 == width &&
   209       src_stride_v * 4 == width &&
   210       dst_stride_argb == width * 4) {
   211     width *= height;
   212     height = 1;
   213     src_stride_y = src_stride_u = src_stride_v = dst_stride_argb = 0;
   214   }
   215   void (*I411ToARGBRow)(const uint8* y_buf,
   216                         const uint8* u_buf,
   217                         const uint8* v_buf,
   218                         uint8* rgb_buf,
   219                         int width) = I411ToARGBRow_C;
   220 #if defined(HAS_I411TOARGBROW_SSSE3)
   221   if (TestCpuFlag(kCpuHasSSSE3) && width >= 8) {
   222     I411ToARGBRow = I411ToARGBRow_Any_SSSE3;
   223     if (IS_ALIGNED(width, 8)) {
   224       I411ToARGBRow = I411ToARGBRow_Unaligned_SSSE3;
   225       if (IS_ALIGNED(dst_argb, 16) && IS_ALIGNED(dst_stride_argb, 16)) {
   226         I411ToARGBRow = I411ToARGBRow_SSSE3;
   227       }
   228     }
   229   }
   230 #elif defined(HAS_I411TOARGBROW_NEON)
   231   if (TestCpuFlag(kCpuHasNEON) && width >= 8) {
   232     I411ToARGBRow = I411ToARGBRow_Any_NEON;
   233     if (IS_ALIGNED(width, 8)) {
   234       I411ToARGBRow = I411ToARGBRow_NEON;
   235     }
   236   }
   237 #endif
   239   for (int y = 0; y < height; ++y) {
   240     I411ToARGBRow(src_y, src_u, src_v, dst_argb, width);
   241     dst_argb += dst_stride_argb;
   242     src_y += src_stride_y;
   243     src_u += src_stride_u;
   244     src_v += src_stride_v;
   245   }
   246   return 0;
   247 }
   249 // Convert I400 to ARGB.
   250 LIBYUV_API
   251 int I400ToARGB_Reference(const uint8* src_y, int src_stride_y,
   252                          uint8* dst_argb, int dst_stride_argb,
   253                          int width, int height) {
   254   if (!src_y || !dst_argb ||
   255       width <= 0 || height == 0) {
   256     return -1;
   257   }
   258   // Negative height means invert the image.
   259   if (height < 0) {
   260     height = -height;
   261     dst_argb = dst_argb + (height - 1) * dst_stride_argb;
   262     dst_stride_argb = -dst_stride_argb;
   263   }
   264   // Coalesce rows.
   265   if (src_stride_y == width &&
   266       dst_stride_argb == width * 4) {
   267     width *= height;
   268     height = 1;
   269     src_stride_y = dst_stride_argb = 0;
   270   }
   271   void (*YToARGBRow)(const uint8* y_buf,
   272                      uint8* rgb_buf,
   273                      int width) = YToARGBRow_C;
   274 #if defined(HAS_YTOARGBROW_SSE2)
   275   if (TestCpuFlag(kCpuHasSSE2) && width >= 8 &&
   276       IS_ALIGNED(dst_argb, 16) && IS_ALIGNED(dst_stride_argb, 16)) {
   277     YToARGBRow = YToARGBRow_Any_SSE2;
   278     if (IS_ALIGNED(width, 8)) {
   279       YToARGBRow = YToARGBRow_SSE2;
   280     }
   281   }
   282 #elif defined(HAS_YTOARGBROW_NEON)
   283   if (TestCpuFlag(kCpuHasNEON) && width >= 8) {
   284     YToARGBRow = YToARGBRow_Any_NEON;
   285     if (IS_ALIGNED(width, 8)) {
   286       YToARGBRow = YToARGBRow_NEON;
   287     }
   288   }
   289 #endif
   291   for (int y = 0; y < height; ++y) {
   292     YToARGBRow(src_y, dst_argb, width);
   293     dst_argb += dst_stride_argb;
   294     src_y += src_stride_y;
   295   }
   296   return 0;
   297 }
   299 // Convert I400 to ARGB.
   300 LIBYUV_API
   301 int I400ToARGB(const uint8* src_y, int src_stride_y,
   302                uint8* dst_argb, int dst_stride_argb,
   303                int width, int height) {
   304   if (!src_y || !dst_argb ||
   305       width <= 0 || height == 0) {
   306     return -1;
   307   }
   308   // Negative height means invert the image.
   309   if (height < 0) {
   310     height = -height;
   311     src_y = src_y + (height - 1) * src_stride_y;
   312     src_stride_y = -src_stride_y;
   313   }
   314   // Coalesce rows.
   315   if (src_stride_y == width &&
   316       dst_stride_argb == width * 4) {
   317     width *= height;
   318     height = 1;
   319     src_stride_y = dst_stride_argb = 0;
   320   }
   321   void (*I400ToARGBRow)(const uint8* src_y, uint8* dst_argb, int pix) =
   322       I400ToARGBRow_C;
   323 #if defined(HAS_I400TOARGBROW_SSE2)
   324   if (TestCpuFlag(kCpuHasSSE2) && width >= 8) {
   325     I400ToARGBRow = I400ToARGBRow_Any_SSE2;
   326     if (IS_ALIGNED(width, 8)) {
   327       I400ToARGBRow = I400ToARGBRow_Unaligned_SSE2;
   328       if (IS_ALIGNED(dst_argb, 16) && IS_ALIGNED(dst_stride_argb, 16)) {
   329         I400ToARGBRow = I400ToARGBRow_SSE2;
   330       }
   331     }
   332   }
   333 #elif defined(HAS_I400TOARGBROW_NEON)
   334   if (TestCpuFlag(kCpuHasNEON) && width >= 8) {
   335     I400ToARGBRow = I400ToARGBRow_Any_NEON;
   336     if (IS_ALIGNED(width, 8)) {
   337       I400ToARGBRow = I400ToARGBRow_NEON;
   338     }
   339   }
   340 #endif
   341   for (int y = 0; y < height; ++y) {
   342     I400ToARGBRow(src_y, dst_argb, width);
   343     src_y += src_stride_y;
   344     dst_argb += dst_stride_argb;
   345   }
   346   return 0;
   347 }
   349 // Shuffle table for converting BGRA to ARGB.
   350 static uvec8 kShuffleMaskBGRAToARGB = {
   351   3u, 2u, 1u, 0u, 7u, 6u, 5u, 4u, 11u, 10u, 9u, 8u, 15u, 14u, 13u, 12u
   352 };
   354 // Shuffle table for converting ABGR to ARGB.
   355 static uvec8 kShuffleMaskABGRToARGB = {
   356   2u, 1u, 0u, 3u, 6u, 5u, 4u, 7u, 10u, 9u, 8u, 11u, 14u, 13u, 12u, 15u
   357 };
   359 // Shuffle table for converting RGBA to ARGB.
   360 static uvec8 kShuffleMaskRGBAToARGB = {
   361   1u, 2u, 3u, 0u, 5u, 6u, 7u, 4u, 9u, 10u, 11u, 8u, 13u, 14u, 15u, 12u
   362 };
   364 // Convert BGRA to ARGB.
   365 LIBYUV_API
   366 int BGRAToARGB(const uint8* src_bgra, int src_stride_bgra,
   367                uint8* dst_argb, int dst_stride_argb,
   368                int width, int height) {
   369   return ARGBShuffle(src_bgra, src_stride_bgra,
   370                      dst_argb, dst_stride_argb,
   371                      (const uint8*)(&kShuffleMaskBGRAToARGB),
   372                      width, height);
   373 }
   375 // Convert ABGR to ARGB.
   376 LIBYUV_API
   377 int ABGRToARGB(const uint8* src_abgr, int src_stride_abgr,
   378                uint8* dst_argb, int dst_stride_argb,
   379                int width, int height) {
   380   return ARGBShuffle(src_abgr, src_stride_abgr,
   381                      dst_argb, dst_stride_argb,
   382                      (const uint8*)(&kShuffleMaskABGRToARGB),
   383                      width, height);
   384 }
   386 // Convert RGBA to ARGB.
   387 LIBYUV_API
   388 int RGBAToARGB(const uint8* src_rgba, int src_stride_rgba,
   389                uint8* dst_argb, int dst_stride_argb,
   390                int width, int height) {
   391   return ARGBShuffle(src_rgba, src_stride_rgba,
   392                      dst_argb, dst_stride_argb,
   393                      (const uint8*)(&kShuffleMaskRGBAToARGB),
   394                      width, height);
   395 }
   397 // Convert RGB24 to ARGB.
   398 LIBYUV_API
   399 int RGB24ToARGB(const uint8* src_rgb24, int src_stride_rgb24,
   400                 uint8* dst_argb, int dst_stride_argb,
   401                 int width, int height) {
   402   if (!src_rgb24 || !dst_argb ||
   403       width <= 0 || height == 0) {
   404     return -1;
   405   }
   406   // Negative height means invert the image.
   407   if (height < 0) {
   408     height = -height;
   409     src_rgb24 = src_rgb24 + (height - 1) * src_stride_rgb24;
   410     src_stride_rgb24 = -src_stride_rgb24;
   411   }
   412   // Coalesce rows.
   413   if (src_stride_rgb24 == width * 3 &&
   414       dst_stride_argb == width * 4) {
   415     width *= height;
   416     height = 1;
   417     src_stride_rgb24 = dst_stride_argb = 0;
   418   }
   419   void (*RGB24ToARGBRow)(const uint8* src_rgb, uint8* dst_argb, int pix) =
   420       RGB24ToARGBRow_C;
   421 #if defined(HAS_RGB24TOARGBROW_SSSE3)
   422   if (TestCpuFlag(kCpuHasSSSE3) && width >= 16 &&
   423       IS_ALIGNED(dst_argb, 16) && IS_ALIGNED(dst_stride_argb, 16)) {
   424     RGB24ToARGBRow = RGB24ToARGBRow_Any_SSSE3;
   425     if (IS_ALIGNED(width, 16)) {
   426       RGB24ToARGBRow = RGB24ToARGBRow_SSSE3;
   427     }
   428   }
   429 #elif defined(HAS_RGB24TOARGBROW_NEON)
   430   if (TestCpuFlag(kCpuHasNEON) && width >= 8) {
   431     RGB24ToARGBRow = RGB24ToARGBRow_Any_NEON;
   432     if (IS_ALIGNED(width, 8)) {
   433       RGB24ToARGBRow = RGB24ToARGBRow_NEON;
   434     }
   435   }
   436 #endif
   438   for (int y = 0; y < height; ++y) {
   439     RGB24ToARGBRow(src_rgb24, dst_argb, width);
   440     src_rgb24 += src_stride_rgb24;
   441     dst_argb += dst_stride_argb;
   442   }
   443   return 0;
   444 }
   446 // Convert RAW to ARGB.
   447 LIBYUV_API
   448 int RAWToARGB(const uint8* src_raw, int src_stride_raw,
   449               uint8* dst_argb, int dst_stride_argb,
   450               int width, int height) {
   451   if (!src_raw || !dst_argb ||
   452       width <= 0 || height == 0) {
   453     return -1;
   454   }
   455   // Negative height means invert the image.
   456   if (height < 0) {
   457     height = -height;
   458     src_raw = src_raw + (height - 1) * src_stride_raw;
   459     src_stride_raw = -src_stride_raw;
   460   }
   461   // Coalesce rows.
   462   if (src_stride_raw == width * 3 &&
   463       dst_stride_argb == width * 4) {
   464     width *= height;
   465     height = 1;
   466     src_stride_raw = dst_stride_argb = 0;
   467   }
   468   void (*RAWToARGBRow)(const uint8* src_rgb, uint8* dst_argb, int pix) =
   469       RAWToARGBRow_C;
   470 #if defined(HAS_RAWTOARGBROW_SSSE3)
   471   if (TestCpuFlag(kCpuHasSSSE3) && width >= 16 &&
   472       IS_ALIGNED(dst_argb, 16) && IS_ALIGNED(dst_stride_argb, 16)) {
   473     RAWToARGBRow = RAWToARGBRow_Any_SSSE3;
   474     if (IS_ALIGNED(width, 16)) {
   475       RAWToARGBRow = RAWToARGBRow_SSSE3;
   476     }
   477   }
   478 #elif defined(HAS_RAWTOARGBROW_NEON)
   479   if (TestCpuFlag(kCpuHasNEON) && width >= 8) {
   480     RAWToARGBRow = RAWToARGBRow_Any_NEON;
   481     if (IS_ALIGNED(width, 8)) {
   482       RAWToARGBRow = RAWToARGBRow_NEON;
   483     }
   484   }
   485 #endif
   487   for (int y = 0; y < height; ++y) {
   488     RAWToARGBRow(src_raw, dst_argb, width);
   489     src_raw += src_stride_raw;
   490     dst_argb += dst_stride_argb;
   491   }
   492   return 0;
   493 }
   495 // Convert RGB565 to ARGB.
   496 LIBYUV_API
   497 int RGB565ToARGB(const uint8* src_rgb565, int src_stride_rgb565,
   498                  uint8* dst_argb, int dst_stride_argb,
   499                  int width, int height) {
   500   if (!src_rgb565 || !dst_argb ||
   501       width <= 0 || height == 0) {
   502     return -1;
   503   }
   504   // Negative height means invert the image.
   505   if (height < 0) {
   506     height = -height;
   507     src_rgb565 = src_rgb565 + (height - 1) * src_stride_rgb565;
   508     src_stride_rgb565 = -src_stride_rgb565;
   509   }
   510   // Coalesce rows.
   511   if (src_stride_rgb565 == width * 2 &&
   512       dst_stride_argb == width * 4) {
   513     width *= height;
   514     height = 1;
   515     src_stride_rgb565 = dst_stride_argb = 0;
   516   }
   517   void (*RGB565ToARGBRow)(const uint8* src_rgb565, uint8* dst_argb, int pix) =
   518       RGB565ToARGBRow_C;
   519 #if defined(HAS_RGB565TOARGBROW_SSE2)
   520   if (TestCpuFlag(kCpuHasSSE2) && width >= 8 &&
   521       IS_ALIGNED(dst_argb, 16) && IS_ALIGNED(dst_stride_argb, 16)) {
   522     RGB565ToARGBRow = RGB565ToARGBRow_Any_SSE2;
   523     if (IS_ALIGNED(width, 8)) {
   524       RGB565ToARGBRow = RGB565ToARGBRow_SSE2;
   525     }
   526   }
   527 #elif defined(HAS_RGB565TOARGBROW_NEON)
   528   if (TestCpuFlag(kCpuHasNEON) && width >= 8) {
   529     RGB565ToARGBRow = RGB565ToARGBRow_Any_NEON;
   530     if (IS_ALIGNED(width, 8)) {
   531       RGB565ToARGBRow = RGB565ToARGBRow_NEON;
   532     }
   533   }
   534 #endif
   536   for (int y = 0; y < height; ++y) {
   537     RGB565ToARGBRow(src_rgb565, dst_argb, width);
   538     src_rgb565 += src_stride_rgb565;
   539     dst_argb += dst_stride_argb;
   540   }
   541   return 0;
   542 }
   544 // Convert ARGB1555 to ARGB.
   545 LIBYUV_API
   546 int ARGB1555ToARGB(const uint8* src_argb1555, int src_stride_argb1555,
   547                    uint8* dst_argb, int dst_stride_argb,
   548                    int width, int height) {
   549   if (!src_argb1555 || !dst_argb ||
   550       width <= 0 || height == 0) {
   551     return -1;
   552   }
   553   // Negative height means invert the image.
   554   if (height < 0) {
   555     height = -height;
   556     src_argb1555 = src_argb1555 + (height - 1) * src_stride_argb1555;
   557     src_stride_argb1555 = -src_stride_argb1555;
   558   }
   559   // Coalesce rows.
   560   if (src_stride_argb1555 == width * 2 &&
   561       dst_stride_argb == width * 4) {
   562     width *= height;
   563     height = 1;
   564     src_stride_argb1555 = dst_stride_argb = 0;
   565   }
   566   void (*ARGB1555ToARGBRow)(const uint8* src_argb1555, uint8* dst_argb,
   567                             int pix) = ARGB1555ToARGBRow_C;
   568 #if defined(HAS_ARGB1555TOARGBROW_SSE2)
   569   if (TestCpuFlag(kCpuHasSSE2) && width >= 8 &&
   570       IS_ALIGNED(dst_argb, 16) && IS_ALIGNED(dst_stride_argb, 16)) {
   571     ARGB1555ToARGBRow = ARGB1555ToARGBRow_Any_SSE2;
   572     if (IS_ALIGNED(width, 8)) {
   573       ARGB1555ToARGBRow = ARGB1555ToARGBRow_SSE2;
   574     }
   575   }
   576 #elif defined(HAS_ARGB1555TOARGBROW_NEON)
   577   if (TestCpuFlag(kCpuHasNEON) && width >= 8) {
   578     ARGB1555ToARGBRow = ARGB1555ToARGBRow_Any_NEON;
   579     if (IS_ALIGNED(width, 8)) {
   580       ARGB1555ToARGBRow = ARGB1555ToARGBRow_NEON;
   581     }
   582   }
   583 #endif
   585   for (int y = 0; y < height; ++y) {
   586     ARGB1555ToARGBRow(src_argb1555, dst_argb, width);
   587     src_argb1555 += src_stride_argb1555;
   588     dst_argb += dst_stride_argb;
   589   }
   590   return 0;
   591 }
   593 // Convert ARGB4444 to ARGB.
   594 LIBYUV_API
   595 int ARGB4444ToARGB(const uint8* src_argb4444, int src_stride_argb4444,
   596                    uint8* dst_argb, int dst_stride_argb,
   597                    int width, int height) {
   598   if (!src_argb4444 || !dst_argb ||
   599       width <= 0 || height == 0) {
   600     return -1;
   601   }
   602   // Negative height means invert the image.
   603   if (height < 0) {
   604     height = -height;
   605     src_argb4444 = src_argb4444 + (height - 1) * src_stride_argb4444;
   606     src_stride_argb4444 = -src_stride_argb4444;
   607   }
   608   // Coalesce rows.
   609   if (src_stride_argb4444 == width * 2 &&
   610       dst_stride_argb == width * 4) {
   611     width *= height;
   612     height = 1;
   613     src_stride_argb4444 = dst_stride_argb = 0;
   614   }
   615   void (*ARGB4444ToARGBRow)(const uint8* src_argb4444, uint8* dst_argb,
   616                             int pix) = ARGB4444ToARGBRow_C;
   617 #if defined(HAS_ARGB4444TOARGBROW_SSE2)
   618   if (TestCpuFlag(kCpuHasSSE2) && width >= 8 &&
   619       IS_ALIGNED(dst_argb, 16) && IS_ALIGNED(dst_stride_argb, 16)) {
   620     ARGB4444ToARGBRow = ARGB4444ToARGBRow_Any_SSE2;
   621     if (IS_ALIGNED(width, 8)) {
   622       ARGB4444ToARGBRow = ARGB4444ToARGBRow_SSE2;
   623     }
   624   }
   625 #elif defined(HAS_ARGB4444TOARGBROW_NEON)
   626   if (TestCpuFlag(kCpuHasNEON) && width >= 8) {
   627     ARGB4444ToARGBRow = ARGB4444ToARGBRow_Any_NEON;
   628     if (IS_ALIGNED(width, 8)) {
   629       ARGB4444ToARGBRow = ARGB4444ToARGBRow_NEON;
   630     }
   631   }
   632 #endif
   634   for (int y = 0; y < height; ++y) {
   635     ARGB4444ToARGBRow(src_argb4444, dst_argb, width);
   636     src_argb4444 += src_stride_argb4444;
   637     dst_argb += dst_stride_argb;
   638   }
   639   return 0;
   640 }
   642 // Convert NV12 to ARGB.
   643 LIBYUV_API
   644 int NV12ToARGB(const uint8* src_y, int src_stride_y,
   645                const uint8* src_uv, int src_stride_uv,
   646                uint8* dst_argb, int dst_stride_argb,
   647                int width, int height) {
   648   if (!src_y || !src_uv || !dst_argb ||
   649       width <= 0 || height == 0) {
   650     return -1;
   651   }
   652   // Negative height means invert the image.
   653   if (height < 0) {
   654     height = -height;
   655     dst_argb = dst_argb + (height - 1) * dst_stride_argb;
   656     dst_stride_argb = -dst_stride_argb;
   657   }
   658   void (*NV12ToARGBRow)(const uint8* y_buf,
   659                         const uint8* uv_buf,
   660                         uint8* rgb_buf,
   661                         int width) = NV12ToARGBRow_C;
   662 #if defined(HAS_NV12TOARGBROW_SSSE3)
   663   if (TestCpuFlag(kCpuHasSSSE3) && width >= 8) {
   664     NV12ToARGBRow = NV12ToARGBRow_Any_SSSE3;
   665     if (IS_ALIGNED(width, 8)) {
   666       NV12ToARGBRow = NV12ToARGBRow_Unaligned_SSSE3;
   667       if (IS_ALIGNED(dst_argb, 16) && IS_ALIGNED(dst_stride_argb, 16)) {
   668         NV12ToARGBRow = NV12ToARGBRow_SSSE3;
   669       }
   670     }
   671   }
   672 #elif defined(HAS_NV12TOARGBROW_NEON)
   673   if (TestCpuFlag(kCpuHasNEON) && width >= 8) {
   674     NV12ToARGBRow = NV12ToARGBRow_Any_NEON;
   675     if (IS_ALIGNED(width, 8)) {
   676       NV12ToARGBRow = NV12ToARGBRow_NEON;
   677     }
   678   }
   679 #endif
   681   for (int y = 0; y < height; ++y) {
   682     NV12ToARGBRow(src_y, src_uv, dst_argb, width);
   683     dst_argb += dst_stride_argb;
   684     src_y += src_stride_y;
   685     if (y & 1) {
   686       src_uv += src_stride_uv;
   687     }
   688   }
   689   return 0;
   690 }
   692 // Convert NV21 to ARGB.
   693 LIBYUV_API
   694 int NV21ToARGB(const uint8* src_y, int src_stride_y,
   695                const uint8* src_uv, int src_stride_uv,
   696                uint8* dst_argb, int dst_stride_argb,
   697                int width, int height) {
   698   if (!src_y || !src_uv || !dst_argb ||
   699       width <= 0 || height == 0) {
   700     return -1;
   701   }
   702   // Negative height means invert the image.
   703   if (height < 0) {
   704     height = -height;
   705     dst_argb = dst_argb + (height - 1) * dst_stride_argb;
   706     dst_stride_argb = -dst_stride_argb;
   707   }
   708   void (*NV21ToARGBRow)(const uint8* y_buf,
   709                         const uint8* uv_buf,
   710                         uint8* rgb_buf,
   711                         int width) = NV21ToARGBRow_C;
   712 #if defined(HAS_NV21TOARGBROW_SSSE3)
   713   if (TestCpuFlag(kCpuHasSSSE3) && width >= 8) {
   714     NV21ToARGBRow = NV21ToARGBRow_Any_SSSE3;
   715     if (IS_ALIGNED(width, 8)) {
   716       NV21ToARGBRow = NV21ToARGBRow_Unaligned_SSSE3;
   717       if (IS_ALIGNED(dst_argb, 16) && IS_ALIGNED(dst_stride_argb, 16)) {
   718         NV21ToARGBRow = NV21ToARGBRow_SSSE3;
   719       }
   720     }
   721   }
   722 #endif
   723 #if defined(HAS_NV21TOARGBROW_NEON)
   724   if (TestCpuFlag(kCpuHasNEON) && width >= 8) {
   725     NV21ToARGBRow = NV21ToARGBRow_Any_NEON;
   726     if (IS_ALIGNED(width, 8)) {
   727       NV21ToARGBRow = NV21ToARGBRow_NEON;
   728     }
   729   }
   730 #endif
   732   for (int y = 0; y < height; ++y) {
   733     NV21ToARGBRow(src_y, src_uv, dst_argb, width);
   734     dst_argb += dst_stride_argb;
   735     src_y += src_stride_y;
   736     if (y & 1) {
   737       src_uv += src_stride_uv;
   738     }
   739   }
   740   return 0;
   741 }
   743 // Convert M420 to ARGB.
   744 LIBYUV_API
   745 int M420ToARGB(const uint8* src_m420, int src_stride_m420,
   746                uint8* dst_argb, int dst_stride_argb,
   747                int width, int height) {
   748   if (!src_m420 || !dst_argb ||
   749       width <= 0 || height == 0) {
   750     return -1;
   751   }
   752   // Negative height means invert the image.
   753   if (height < 0) {
   754     height = -height;
   755     dst_argb = dst_argb + (height - 1) * dst_stride_argb;
   756     dst_stride_argb = -dst_stride_argb;
   757   }
   758   void (*NV12ToARGBRow)(const uint8* y_buf,
   759                         const uint8* uv_buf,
   760                         uint8* rgb_buf,
   761                         int width) = NV12ToARGBRow_C;
   762 #if defined(HAS_NV12TOARGBROW_SSSE3)
   763   if (TestCpuFlag(kCpuHasSSSE3) && width >= 8) {
   764     NV12ToARGBRow = NV12ToARGBRow_Any_SSSE3;
   765     if (IS_ALIGNED(width, 8)) {
   766       NV12ToARGBRow = NV12ToARGBRow_Unaligned_SSSE3;
   767       if (IS_ALIGNED(dst_argb, 16) && IS_ALIGNED(dst_stride_argb, 16)) {
   768         NV12ToARGBRow = NV12ToARGBRow_SSSE3;
   769       }
   770     }
   771   }
   772 #elif defined(HAS_NV12TOARGBROW_NEON)
   773   if (TestCpuFlag(kCpuHasNEON) && width >= 8) {
   774     NV12ToARGBRow = NV12ToARGBRow_Any_NEON;
   775     if (IS_ALIGNED(width, 8)) {
   776       NV12ToARGBRow = NV12ToARGBRow_NEON;
   777     }
   778   }
   779 #endif
   781   for (int y = 0; y < height - 1; y += 2) {
   782     NV12ToARGBRow(src_m420, src_m420 + src_stride_m420 * 2, dst_argb, width);
   783     NV12ToARGBRow(src_m420 + src_stride_m420, src_m420 + src_stride_m420 * 2,
   784                   dst_argb + dst_stride_argb, width);
   785     dst_argb += dst_stride_argb * 2;
   786     src_m420 += src_stride_m420 * 3;
   787   }
   788   if (height & 1) {
   789     NV12ToARGBRow(src_m420, src_m420 + src_stride_m420 * 2, dst_argb, width);
   790   }
   791   return 0;
   792 }
   794 // Convert YUY2 to ARGB.
   795 LIBYUV_API
   796 int YUY2ToARGB(const uint8* src_yuy2, int src_stride_yuy2,
   797                uint8* dst_argb, int dst_stride_argb,
   798                int width, int height) {
   799   if (!src_yuy2 || !dst_argb ||
   800       width <= 0 || height == 0) {
   801     return -1;
   802   }
   803   // Negative height means invert the image.
   804   if (height < 0) {
   805     height = -height;
   806     src_yuy2 = src_yuy2 + (height - 1) * src_stride_yuy2;
   807     src_stride_yuy2 = -src_stride_yuy2;
   808   }
   809   // Coalesce rows.
   810   if (src_stride_yuy2 == width * 2 &&
   811       dst_stride_argb == width * 4) {
   812     width *= height;
   813     height = 1;
   814     src_stride_yuy2 = dst_stride_argb = 0;
   815   }
   816   void (*YUY2ToARGBRow)(const uint8* src_yuy2, uint8* dst_argb, int pix) =
   817       YUY2ToARGBRow_C;
   818 #if defined(HAS_YUY2TOARGBROW_SSSE3)
   819   // Posix is 16, Windows is 8.
   820   if (TestCpuFlag(kCpuHasSSSE3) && width >= 16) {
   821     YUY2ToARGBRow = YUY2ToARGBRow_Any_SSSE3;
   822     if (IS_ALIGNED(width, 16)) {
   823       YUY2ToARGBRow = YUY2ToARGBRow_Unaligned_SSSE3;
   824       if (IS_ALIGNED(src_yuy2, 16) && IS_ALIGNED(src_stride_yuy2, 16) &&
   825           IS_ALIGNED(dst_argb, 16) && IS_ALIGNED(dst_stride_argb, 16)) {
   826         YUY2ToARGBRow = YUY2ToARGBRow_SSSE3;
   827       }
   828     }
   829   }
   830 #elif defined(HAS_YUY2TOARGBROW_NEON)
   831   if (TestCpuFlag(kCpuHasNEON) && width >= 8) {
   832     YUY2ToARGBRow = YUY2ToARGBRow_Any_NEON;
   833     if (IS_ALIGNED(width, 8)) {
   834       YUY2ToARGBRow = YUY2ToARGBRow_NEON;
   835     }
   836   }
   837 #endif
   838   for (int y = 0; y < height; ++y) {
   839     YUY2ToARGBRow(src_yuy2, dst_argb, width);
   840     src_yuy2 += src_stride_yuy2;
   841     dst_argb += dst_stride_argb;
   842   }
   843   return 0;
   844 }
   846 // Convert UYVY to ARGB.
   847 LIBYUV_API
   848 int UYVYToARGB(const uint8* src_uyvy, int src_stride_uyvy,
   849                uint8* dst_argb, int dst_stride_argb,
   850                int width, int height) {
   851   if (!src_uyvy || !dst_argb ||
   852       width <= 0 || height == 0) {
   853     return -1;
   854   }
   855   // Negative height means invert the image.
   856   if (height < 0) {
   857     height = -height;
   858     src_uyvy = src_uyvy + (height - 1) * src_stride_uyvy;
   859     src_stride_uyvy = -src_stride_uyvy;
   860   }
   861   // Coalesce rows.
   862   if (src_stride_uyvy == width * 2 &&
   863       dst_stride_argb == width * 4) {
   864     width *= height;
   865     height = 1;
   866     src_stride_uyvy = dst_stride_argb = 0;
   867   }
   868   void (*UYVYToARGBRow)(const uint8* src_uyvy, uint8* dst_argb, int pix) =
   869       UYVYToARGBRow_C;
   870 #if defined(HAS_UYVYTOARGBROW_SSSE3)
   871   // Posix is 16, Windows is 8.
   872   if (TestCpuFlag(kCpuHasSSSE3) && width >= 16) {
   873     UYVYToARGBRow = UYVYToARGBRow_Any_SSSE3;
   874     if (IS_ALIGNED(width, 16)) {
   875       UYVYToARGBRow = UYVYToARGBRow_Unaligned_SSSE3;
   876       if (IS_ALIGNED(src_uyvy, 16) && IS_ALIGNED(src_stride_uyvy, 16) &&
   877           IS_ALIGNED(dst_argb, 16) && IS_ALIGNED(dst_stride_argb, 16)) {
   878         UYVYToARGBRow = UYVYToARGBRow_SSSE3;
   879       }
   880     }
   881   }
   882 #elif defined(HAS_UYVYTOARGBROW_NEON)
   883   if (TestCpuFlag(kCpuHasNEON) && width >= 8) {
   884     UYVYToARGBRow = UYVYToARGBRow_Any_NEON;
   885     if (IS_ALIGNED(width, 8)) {
   886       UYVYToARGBRow = UYVYToARGBRow_NEON;
   887     }
   888   }
   889 #endif
   890   for (int y = 0; y < height; ++y) {
   891     UYVYToARGBRow(src_uyvy, dst_argb, width);
   892     src_uyvy += src_stride_uyvy;
   893     dst_argb += dst_stride_argb;
   894   }
   895   return 0;
   896 }
   898 #ifdef __cplusplus
   899 }  // extern "C"
   900 }  // namespace libyuv
   901 #endif

mercurial