Home | History | Annotate | Download | only in pixman
      1 /*
      2  * Copyright  2010 Nokia Corporation
      3  *
      4  * Permission is hereby granted, free of charge, to any person obtaining a
      5  * copy of this software and associated documentation files (the "Software"),
      6  * to deal in the Software without restriction, including without limitation
      7  * the rights to use, copy, modify, merge, publish, distribute, sublicense,
      8  * and/or sell copies of the Software, and to permit persons to whom the
      9  * Software is furnished to do so, subject to the following conditions:
     10  *
     11  * The above copyright notice and this permission notice (including the next
     12  * paragraph) shall be included in all copies or substantial portions of the
     13  * Software.
     14  *
     15  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
     16  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
     17  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
     18  * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
     19  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
     20  * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
     21  * DEALINGS IN THE SOFTWARE.
     22  *
     23  * Author:  Siarhei Siamashka (siarhei.siamashka (at) nokia.com)
     24  */
     25 
     26 #ifndef PIXMAN_ARM_COMMON_H
     27 #define PIXMAN_ARM_COMMON_H
     28 
     29 #include "pixman-inlines.h"
     30 
     31 /* Define some macros which can expand into proxy functions between
     32  * ARM assembly optimized functions and the rest of pixman fast path API.
     33  *
     34  * All the low level ARM assembly functions have to use ARM EABI
     35  * calling convention and take up to 8 arguments:
     36  *    width, height, dst, dst_stride, src, src_stride, mask, mask_stride
     37  *
     38  * The arguments are ordered with the most important coming first (the
     39  * first 4 arguments are passed to function in registers, the rest are
     40  * on stack). The last arguments are optional, for example if the
     41  * function is not using mask, then 'mask' and 'mask_stride' can be
     42  * omitted when doing a function call.
     43  *
     44  * Arguments 'src' and 'mask' contain either a pointer to the top left
     45  * pixel of the composited rectangle or a pixel color value depending
     46  * on the function type. In the case of just a color value (solid source
     47  * or mask), the corresponding stride argument is unused.
     48  */
     49 
     50 #define SKIP_ZERO_SRC  1
     51 #define SKIP_ZERO_MASK 2
     52 
     53 #define PIXMAN_ARM_BIND_FAST_PATH_SRC_DST(cputype, name,                \
     54                                           src_type, src_cnt,            \
     55                                           dst_type, dst_cnt)            \
     56 void                                                                    \
     57 pixman_composite_##name##_asm_##cputype (int32_t   w,                   \
     58                                          int32_t   h,                   \
     59                                          dst_type *dst,                 \
     60                                          int32_t   dst_stride,          \
     61                                          src_type *src,                 \
     62                                          int32_t   src_stride);         \
     63                                                                         \
     64 static void                                                             \
     65 cputype##_composite_##name (pixman_implementation_t *imp,               \
     66                             pixman_composite_info_t *info)              \
     67 {                                                                       \
     68     PIXMAN_COMPOSITE_ARGS (info);                                       \
     69     dst_type *dst_line;							\
     70     src_type *src_line;                                                 \
     71     int32_t dst_stride, src_stride;                                     \
     72                                                                         \
     73     PIXMAN_IMAGE_GET_LINE (src_image, src_x, src_y, src_type,           \
     74                            src_stride, src_line, src_cnt);              \
     75     PIXMAN_IMAGE_GET_LINE (dest_image, dest_x, dest_y, dst_type,        \
     76                            dst_stride, dst_line, dst_cnt);              \
     77                                                                         \
     78     pixman_composite_##name##_asm_##cputype (width, height,             \
     79                                              dst_line, dst_stride,      \
     80                                              src_line, src_stride);     \
     81 }
     82 
     83 #define PIXMAN_ARM_BIND_FAST_PATH_N_DST(flags, cputype, name,           \
     84                                         dst_type, dst_cnt)              \
     85 void                                                                    \
     86 pixman_composite_##name##_asm_##cputype (int32_t    w,                  \
     87                                          int32_t    h,                  \
     88                                          dst_type  *dst,                \
     89                                          int32_t    dst_stride,         \
     90                                          uint32_t   src);               \
     91                                                                         \
     92 static void                                                             \
     93 cputype##_composite_##name (pixman_implementation_t *imp,               \
     94 			    pixman_composite_info_t *info)              \
     95 {                                                                       \
     96     PIXMAN_COMPOSITE_ARGS (info);					\
     97     dst_type  *dst_line;                                                \
     98     int32_t    dst_stride;                                              \
     99     uint32_t   src;                                                     \
    100                                                                         \
    101     src = _pixman_image_get_solid (					\
    102 	imp, src_image, dest_image->bits.format);			\
    103                                                                         \
    104     if ((flags & SKIP_ZERO_SRC) && src == 0)                            \
    105 	return;                                                         \
    106                                                                         \
    107     PIXMAN_IMAGE_GET_LINE (dest_image, dest_x, dest_y, dst_type,        \
    108                            dst_stride, dst_line, dst_cnt);              \
    109                                                                         \
    110     pixman_composite_##name##_asm_##cputype (width, height,             \
    111                                              dst_line, dst_stride,      \
    112                                              src);                      \
    113 }
    114 
    115 #define PIXMAN_ARM_BIND_FAST_PATH_N_MASK_DST(flags, cputype, name,      \
    116                                              mask_type, mask_cnt,       \
    117                                              dst_type, dst_cnt)         \
    118 void                                                                    \
    119 pixman_composite_##name##_asm_##cputype (int32_t    w,                  \
    120                                          int32_t    h,                  \
    121                                          dst_type  *dst,                \
    122                                          int32_t    dst_stride,         \
    123                                          uint32_t   src,                \
    124                                          int32_t    unused,             \
    125                                          mask_type *mask,               \
    126                                          int32_t    mask_stride);       \
    127                                                                         \
    128 static void                                                             \
    129 cputype##_composite_##name (pixman_implementation_t *imp,               \
    130                             pixman_composite_info_t *info)              \
    131 {                                                                       \
    132     PIXMAN_COMPOSITE_ARGS (info);                                       \
    133     dst_type  *dst_line;						\
    134     mask_type *mask_line;                                               \
    135     int32_t    dst_stride, mask_stride;                                 \
    136     uint32_t   src;                                                     \
    137                                                                         \
    138     src = _pixman_image_get_solid (					\
    139 	imp, src_image, dest_image->bits.format);			\
    140                                                                         \
    141     if ((flags & SKIP_ZERO_SRC) && src == 0)                            \
    142 	return;                                                         \
    143                                                                         \
    144     PIXMAN_IMAGE_GET_LINE (dest_image, dest_x, dest_y, dst_type,        \
    145                            dst_stride, dst_line, dst_cnt);              \
    146     PIXMAN_IMAGE_GET_LINE (mask_image, mask_x, mask_y, mask_type,       \
    147                            mask_stride, mask_line, mask_cnt);           \
    148                                                                         \
    149     pixman_composite_##name##_asm_##cputype (width, height,             \
    150                                              dst_line, dst_stride,      \
    151                                              src, 0,                    \
    152                                              mask_line, mask_stride);   \
    153 }
    154 
    155 #define PIXMAN_ARM_BIND_FAST_PATH_SRC_N_DST(flags, cputype, name,       \
    156                                             src_type, src_cnt,          \
    157                                             dst_type, dst_cnt)          \
    158 void                                                                    \
    159 pixman_composite_##name##_asm_##cputype (int32_t    w,                  \
    160                                          int32_t    h,                  \
    161                                          dst_type  *dst,                \
    162                                          int32_t    dst_stride,         \
    163                                          src_type  *src,                \
    164                                          int32_t    src_stride,         \
    165                                          uint32_t   mask);              \
    166                                                                         \
    167 static void                                                             \
    168 cputype##_composite_##name (pixman_implementation_t *imp,               \
    169                             pixman_composite_info_t *info)              \
    170 {                                                                       \
    171     PIXMAN_COMPOSITE_ARGS (info);                                       \
    172     dst_type  *dst_line;						\
    173     src_type  *src_line;                                                \
    174     int32_t    dst_stride, src_stride;                                  \
    175     uint32_t   mask;                                                    \
    176                                                                         \
    177     mask = _pixman_image_get_solid (					\
    178 	imp, mask_image, dest_image->bits.format);			\
    179                                                                         \
    180     if ((flags & SKIP_ZERO_MASK) && mask == 0)                          \
    181 	return;                                                         \
    182                                                                         \
    183     PIXMAN_IMAGE_GET_LINE (dest_image, dest_x, dest_y, dst_type,        \
    184                            dst_stride, dst_line, dst_cnt);              \
    185     PIXMAN_IMAGE_GET_LINE (src_image, src_x, src_y, src_type,           \
    186                            src_stride, src_line, src_cnt);              \
    187                                                                         \
    188     pixman_composite_##name##_asm_##cputype (width, height,             \
    189                                              dst_line, dst_stride,      \
    190                                              src_line, src_stride,      \
    191                                              mask);                     \
    192 }
    193 
    194 #define PIXMAN_ARM_BIND_FAST_PATH_SRC_MASK_DST(cputype, name,           \
    195                                                src_type, src_cnt,       \
    196                                                mask_type, mask_cnt,     \
    197                                                dst_type, dst_cnt)       \
    198 void                                                                    \
    199 pixman_composite_##name##_asm_##cputype (int32_t    w,                  \
    200                                          int32_t    h,                  \
    201                                          dst_type  *dst,                \
    202                                          int32_t    dst_stride,         \
    203                                          src_type  *src,                \
    204                                          int32_t    src_stride,         \
    205                                          mask_type *mask,               \
    206                                          int32_t    mask_stride);       \
    207                                                                         \
    208 static void                                                             \
    209 cputype##_composite_##name (pixman_implementation_t *imp,               \
    210                             pixman_composite_info_t *info)              \
    211 {                                                                       \
    212     PIXMAN_COMPOSITE_ARGS (info);                                       \
    213     dst_type  *dst_line;						\
    214     src_type  *src_line;                                                \
    215     mask_type *mask_line;                                               \
    216     int32_t    dst_stride, src_stride, mask_stride;                     \
    217                                                                         \
    218     PIXMAN_IMAGE_GET_LINE (dest_image, dest_x, dest_y, dst_type,        \
    219                            dst_stride, dst_line, dst_cnt);              \
    220     PIXMAN_IMAGE_GET_LINE (src_image, src_x, src_y, src_type,           \
    221                            src_stride, src_line, src_cnt);              \
    222     PIXMAN_IMAGE_GET_LINE (mask_image, mask_x, mask_y, mask_type,       \
    223                            mask_stride, mask_line, mask_cnt);           \
    224                                                                         \
    225     pixman_composite_##name##_asm_##cputype (width, height,             \
    226                                              dst_line, dst_stride,      \
    227                                              src_line, src_stride,      \
    228                                              mask_line, mask_stride);   \
    229 }
    230 
    231 #define PIXMAN_ARM_BIND_SCALED_NEAREST_SRC_DST(cputype, name, op,             \
    232                                                src_type, dst_type)            \
    233 void                                                                          \
    234 pixman_scaled_nearest_scanline_##name##_##op##_asm_##cputype (                \
    235                                                    int32_t          w,        \
    236                                                    dst_type *       dst,      \
    237                                                    const src_type * src,      \
    238                                                    pixman_fixed_t   vx,       \
    239                                                    pixman_fixed_t   unit_x,   \
    240                                                    pixman_fixed_t   max_vx);  \
    241                                                                               \
    242 static force_inline void                                                      \
    243 scaled_nearest_scanline_##cputype##_##name##_##op (dst_type *       pd,       \
    244                                                    const src_type * ps,       \
    245                                                    int32_t          w,        \
    246                                                    pixman_fixed_t   vx,       \
    247                                                    pixman_fixed_t   unit_x,   \
    248                                                    pixman_fixed_t   max_vx,   \
    249                                                    pixman_bool_t    zero_src) \
    250 {                                                                             \
    251     pixman_scaled_nearest_scanline_##name##_##op##_asm_##cputype (w, pd, ps,  \
    252                                                                   vx, unit_x, \
    253                                                                   max_vx);    \
    254 }                                                                             \
    255                                                                               \
    256 FAST_NEAREST_MAINLOOP (cputype##_##name##_cover_##op,                         \
    257                        scaled_nearest_scanline_##cputype##_##name##_##op,     \
    258                        src_type, dst_type, COVER)                             \
    259 FAST_NEAREST_MAINLOOP (cputype##_##name##_none_##op,                          \
    260                        scaled_nearest_scanline_##cputype##_##name##_##op,     \
    261                        src_type, dst_type, NONE)                              \
    262 FAST_NEAREST_MAINLOOP (cputype##_##name##_pad_##op,                           \
    263                        scaled_nearest_scanline_##cputype##_##name##_##op,     \
    264                        src_type, dst_type, PAD)                               \
    265 FAST_NEAREST_MAINLOOP (cputype##_##name##_normal_##op,                        \
    266                        scaled_nearest_scanline_##cputype##_##name##_##op,     \
    267                        src_type, dst_type, NORMAL)
    268 
    269 /* Provide entries for the fast path table */
    270 #define PIXMAN_ARM_SIMPLE_NEAREST_FAST_PATH(op,s,d,func)                      \
    271     SIMPLE_NEAREST_FAST_PATH_COVER (op,s,d,func),                             \
    272     SIMPLE_NEAREST_FAST_PATH_NONE (op,s,d,func),                              \
    273     SIMPLE_NEAREST_FAST_PATH_PAD (op,s,d,func),                               \
    274     SIMPLE_NEAREST_FAST_PATH_NORMAL (op,s,d,func)
    275 
    276 #define PIXMAN_ARM_BIND_SCALED_NEAREST_SRC_A8_DST(flags, cputype, name, op,   \
    277                                                   src_type, dst_type)         \
    278 void                                                                          \
    279 pixman_scaled_nearest_scanline_##name##_##op##_asm_##cputype (                \
    280                                                    int32_t          w,        \
    281                                                    dst_type *       dst,      \
    282                                                    const src_type * src,      \
    283                                                    pixman_fixed_t   vx,       \
    284                                                    pixman_fixed_t   unit_x,   \
    285                                                    pixman_fixed_t   max_vx,   \
    286                                                    const uint8_t *  mask);    \
    287                                                                               \
    288 static force_inline void                                                      \
    289 scaled_nearest_scanline_##cputype##_##name##_##op (const uint8_t *  mask,     \
    290                                                    dst_type *       pd,       \
    291                                                    const src_type * ps,       \
    292                                                    int32_t          w,        \
    293                                                    pixman_fixed_t   vx,       \
    294                                                    pixman_fixed_t   unit_x,   \
    295                                                    pixman_fixed_t   max_vx,   \
    296                                                    pixman_bool_t    zero_src) \
    297 {                                                                             \
    298     if ((flags & SKIP_ZERO_SRC) && zero_src)                                  \
    299 	return;                                                               \
    300     pixman_scaled_nearest_scanline_##name##_##op##_asm_##cputype (w, pd, ps,  \
    301                                                                   vx, unit_x, \
    302                                                                   max_vx,     \
    303                                                                   mask);      \
    304 }                                                                             \
    305                                                                               \
    306 FAST_NEAREST_MAINLOOP_COMMON (cputype##_##name##_cover_##op,                  \
    307                               scaled_nearest_scanline_##cputype##_##name##_##op,\
    308                               src_type, uint8_t, dst_type, COVER, TRUE, FALSE)\
    309 FAST_NEAREST_MAINLOOP_COMMON (cputype##_##name##_none_##op,                   \
    310                               scaled_nearest_scanline_##cputype##_##name##_##op,\
    311                               src_type, uint8_t, dst_type, NONE, TRUE, FALSE) \
    312 FAST_NEAREST_MAINLOOP_COMMON (cputype##_##name##_pad_##op,                    \
    313                               scaled_nearest_scanline_##cputype##_##name##_##op,\
    314                               src_type, uint8_t, dst_type, PAD, TRUE, FALSE)  \
    315 FAST_NEAREST_MAINLOOP_COMMON (cputype##_##name##_normal_##op,                 \
    316                               scaled_nearest_scanline_##cputype##_##name##_##op,\
    317                               src_type, uint8_t, dst_type, NORMAL, TRUE, FALSE)
    318 
    319 /* Provide entries for the fast path table */
    320 #define PIXMAN_ARM_SIMPLE_NEAREST_A8_MASK_FAST_PATH(op,s,d,func)              \
    321     SIMPLE_NEAREST_A8_MASK_FAST_PATH_COVER (op,s,d,func),                     \
    322     SIMPLE_NEAREST_A8_MASK_FAST_PATH_NONE (op,s,d,func),                      \
    323     SIMPLE_NEAREST_A8_MASK_FAST_PATH_PAD (op,s,d,func),                       \
    324     SIMPLE_NEAREST_A8_MASK_FAST_PATH_NORMAL (op,s,d,func)
    325 
    326 /*****************************************************************************/
    327 
    328 #define PIXMAN_ARM_BIND_SCALED_BILINEAR_SRC_DST(flags, cputype, name, op,     \
    329                                                 src_type, dst_type)           \
    330 void                                                                          \
    331 pixman_scaled_bilinear_scanline_##name##_##op##_asm_##cputype (               \
    332                                                 dst_type *       dst,         \
    333                                                 const src_type * top,         \
    334                                                 const src_type * bottom,      \
    335                                                 int              wt,          \
    336                                                 int              wb,          \
    337                                                 pixman_fixed_t   x,           \
    338                                                 pixman_fixed_t   ux,          \
    339                                                 int              width);      \
    340                                                                               \
    341 static force_inline void                                                      \
    342 scaled_bilinear_scanline_##cputype##_##name##_##op (                          \
    343                                                 dst_type *       dst,         \
    344                                                 const uint32_t * mask,        \
    345                                                 const src_type * src_top,     \
    346                                                 const src_type * src_bottom,  \
    347                                                 int32_t          w,           \
    348                                                 int              wt,          \
    349                                                 int              wb,          \
    350                                                 pixman_fixed_t   vx,          \
    351                                                 pixman_fixed_t   unit_x,      \
    352                                                 pixman_fixed_t   max_vx,      \
    353                                                 pixman_bool_t    zero_src)    \
    354 {                                                                             \
    355     if ((flags & SKIP_ZERO_SRC) && zero_src)                                  \
    356 	return;                                                               \
    357     pixman_scaled_bilinear_scanline_##name##_##op##_asm_##cputype (           \
    358                             dst, src_top, src_bottom, wt, wb, vx, unit_x, w); \
    359 }                                                                             \
    360                                                                               \
    361 FAST_BILINEAR_MAINLOOP_COMMON (cputype##_##name##_cover_##op,                 \
    362                        scaled_bilinear_scanline_##cputype##_##name##_##op,    \
    363                        src_type, uint32_t, dst_type, COVER, FLAG_NONE)        \
    364 FAST_BILINEAR_MAINLOOP_COMMON (cputype##_##name##_none_##op,                  \
    365                        scaled_bilinear_scanline_##cputype##_##name##_##op,    \
    366                        src_type, uint32_t, dst_type, NONE, FLAG_NONE)         \
    367 FAST_BILINEAR_MAINLOOP_COMMON (cputype##_##name##_pad_##op,                   \
    368                        scaled_bilinear_scanline_##cputype##_##name##_##op,    \
    369                        src_type, uint32_t, dst_type, PAD, FLAG_NONE)          \
    370 FAST_BILINEAR_MAINLOOP_COMMON (cputype##_##name##_normal_##op,                \
    371                        scaled_bilinear_scanline_##cputype##_##name##_##op,    \
    372                        src_type, uint32_t, dst_type, NORMAL,                  \
    373                        FLAG_NONE)
    374 
    375 
    376 #define PIXMAN_ARM_BIND_SCALED_BILINEAR_SRC_A8_DST(flags, cputype, name, op,  \
    377                                                 src_type, dst_type)           \
    378 void                                                                          \
    379 pixman_scaled_bilinear_scanline_##name##_##op##_asm_##cputype (               \
    380                                                 dst_type *       dst,         \
    381                                                 const uint8_t *  mask,        \
    382                                                 const src_type * top,         \
    383                                                 const src_type * bottom,      \
    384                                                 int              wt,          \
    385                                                 int              wb,          \
    386                                                 pixman_fixed_t   x,           \
    387                                                 pixman_fixed_t   ux,          \
    388                                                 int              width);      \
    389                                                                               \
    390 static force_inline void                                                      \
    391 scaled_bilinear_scanline_##cputype##_##name##_##op (                          \
    392                                                 dst_type *       dst,         \
    393                                                 const uint8_t *  mask,        \
    394                                                 const src_type * src_top,     \
    395                                                 const src_type * src_bottom,  \
    396                                                 int32_t          w,           \
    397                                                 int              wt,          \
    398                                                 int              wb,          \
    399                                                 pixman_fixed_t   vx,          \
    400                                                 pixman_fixed_t   unit_x,      \
    401                                                 pixman_fixed_t   max_vx,      \
    402                                                 pixman_bool_t    zero_src)    \
    403 {                                                                             \
    404     if ((flags & SKIP_ZERO_SRC) && zero_src)                                  \
    405 	return;                                                                   \
    406     pixman_scaled_bilinear_scanline_##name##_##op##_asm_##cputype (           \
    407                       dst, mask, src_top, src_bottom, wt, wb, vx, unit_x, w); \
    408 }                                                                             \
    409                                                                               \
    410 FAST_BILINEAR_MAINLOOP_COMMON (cputype##_##name##_cover_##op,                 \
    411                        scaled_bilinear_scanline_##cputype##_##name##_##op,    \
    412                        src_type, uint8_t, dst_type, COVER,                    \
    413                        FLAG_HAVE_NON_SOLID_MASK)                              \
    414 FAST_BILINEAR_MAINLOOP_COMMON (cputype##_##name##_none_##op,                  \
    415                        scaled_bilinear_scanline_##cputype##_##name##_##op,    \
    416                        src_type, uint8_t, dst_type, NONE,                     \
    417                        FLAG_HAVE_NON_SOLID_MASK)                              \
    418 FAST_BILINEAR_MAINLOOP_COMMON (cputype##_##name##_pad_##op,                   \
    419                        scaled_bilinear_scanline_##cputype##_##name##_##op,    \
    420                        src_type, uint8_t, dst_type, PAD,                      \
    421                        FLAG_HAVE_NON_SOLID_MASK)                              \
    422 FAST_BILINEAR_MAINLOOP_COMMON (cputype##_##name##_normal_##op,                \
    423                        scaled_bilinear_scanline_##cputype##_##name##_##op,    \
    424                        src_type, uint8_t, dst_type, NORMAL,                   \
    425                        FLAG_HAVE_NON_SOLID_MASK)
    426 
    427 
    428 #endif
    429