• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /* Copyright (C) 2007-2008 The Android Open Source Project
2 **
3 ** This software is licensed under the terms of the GNU General Public
4 ** License version 2, as published by the Free Software Foundation, and
5 ** may be copied, distributed, and modified under those terms.
6 **
7 ** This program is distributed in the hope that it will be useful,
8 ** but WITHOUT ANY WARRANTY; without even the implied warranty of
9 ** MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
10 ** GNU General Public License for more details.
11 */
12 #include "android/skin/surface.h"
13 #include "android/skin/argb.h"
14 #include <SDL.h>
15 
16 #define  DEBUG  1
17 
18 #if DEBUG
19 #include "android/utils/debug.h"
20 #define  D(...)   VERBOSE_PRINT(surface,__VA_ARGS__)
21 #else
22 #define  D(...)   ((void)0)
23 #endif
24 
25 struct SkinSurface {
26     int                  refcount;
27     uint32_t*            pixels;
28     SDL_Surface*         surface;
29     SkinSurfaceDoneFunc  done_func;
30     void*                done_user;
31 };
32 
33 static void
skin_surface_free(SkinSurface * s)34 skin_surface_free( SkinSurface*  s )
35 {
36     if (s->done_func) {
37         s->done_func( s->done_user );
38         s->done_func = NULL;
39     }
40     if (s->surface) {
41         SDL_FreeSurface(s->surface);
42         s->surface = NULL;
43     }
44     free(s);
45 }
46 
47 extern SkinSurface*
skin_surface_ref(SkinSurface * surface)48 skin_surface_ref( SkinSurface*  surface )
49 {
50     if (surface)
51         surface->refcount += 1;
52     return surface;
53 }
54 
55 extern void
skin_surface_unrefp(SkinSurface ** psurface)56 skin_surface_unrefp( SkinSurface*  *psurface )
57 {
58     SkinSurface*  surf = *psurface;
59     if (surf) {
60         if (--surf->refcount <= 0)
61             skin_surface_free(surf);
62         *psurface = NULL;
63     }
64 }
65 
66 
67 void
skin_surface_set_done(SkinSurface * s,SkinSurfaceDoneFunc done_func,void * done_user)68 skin_surface_set_done( SkinSurface*  s, SkinSurfaceDoneFunc  done_func, void*  done_user )
69 {
70     s->done_func = done_func;
71     s->done_user = done_user;
72 }
73 
74 #if SDL_BYTEORDER == SDL_BIG_ENDIAN
75 #  define  ARGB32_R_MASK  0xff000000
76 #  define  ARGB32_G_MASK  0x00ff0000
77 #  define  ARGB32_B_MASK  0x0000ff00
78 #  define  ARGB32_A_MASK  0x000000ff
79 #else
80 #  define  ARGB32_R_MASK  0x000000ff
81 #  define  ARGB32_G_MASK  0x0000ff00
82 #  define  ARGB32_B_MASK  0x00ff0000
83 #  define  ARGB32_A_MASK  0xff000000
84 #endif
85 
86 static SDL_Surface*
_sdl_surface_create_rgb(int width,int height,int depth,int flags)87 _sdl_surface_create_rgb( int  width,
88                          int  height,
89                          int  depth,
90                          int  flags )
91 {
92    Uint32   rmask, gmask, bmask, amask;
93 
94     if (depth == 8) {
95         rmask = gmask = bmask = 0;
96         amask = 0xff;
97     } else if (depth == 32) {
98         rmask = ARGB32_R_MASK;
99         gmask = ARGB32_G_MASK;
100         bmask = ARGB32_B_MASK;
101         amask = ARGB32_A_MASK;
102     } else
103         return NULL;
104 
105     return SDL_CreateRGBSurface( flags, width, height, depth,
106                                  rmask, gmask, bmask, amask );
107 }
108 
109 
110 static SDL_Surface*
_sdl_surface_create_rgb_from(int width,int height,int pitch,void * pixels,int depth)111 _sdl_surface_create_rgb_from( int   width,
112                               int   height,
113                               int   pitch,
114                               void* pixels,
115                               int   depth )
116 {
117    Uint32   rmask, gmask, bmask, amask;
118 
119     if (depth == 8) {
120         rmask = gmask = bmask = 0;
121         amask = 0xff;
122     } else if (depth == 32) {
123         rmask = ARGB32_R_MASK;
124         gmask = ARGB32_G_MASK;
125         bmask = ARGB32_B_MASK;
126         amask = ARGB32_A_MASK;
127     } else
128         return NULL;
129 
130     return SDL_CreateRGBSurfaceFrom( pixels, width, height, pitch, depth,
131                                      rmask, gmask, bmask, amask );
132 }
133 
134 
135 static SkinSurface*
_skin_surface_create(SDL_Surface * surface,void * pixels)136 _skin_surface_create( SDL_Surface*  surface,
137                       void*         pixels )
138 {
139     SkinSurface*  s = malloc(sizeof(*s));
140     if (s != NULL) {
141         s->refcount = 1;
142         s->pixels   = pixels;
143         s->surface  = surface;
144         s->done_func = NULL;
145         s->done_user = NULL;
146     }
147     else {
148         SDL_FreeSurface(surface);
149         free(pixels);
150         D( "not enough memory to allocate new skin surface !" );
151     }
152     return  s;
153 }
154 
155 
156 SkinSurface*
skin_surface_create_fast(int w,int h)157 skin_surface_create_fast( int  w, int  h )
158 {
159     SDL_Surface*  surface;
160 
161     surface = _sdl_surface_create_rgb( w, h, 32, SDL_HWSURFACE );
162     if (surface == NULL) {
163         surface = _sdl_surface_create_rgb( w, h, 32, SDL_SWSURFACE );
164         if (surface == NULL) {
165             D( "could not create fast %dx%d ARGB32 surface: %s",
166                w, h, SDL_GetError() );
167             return NULL;
168         }
169     }
170     return _skin_surface_create( surface, NULL );
171 }
172 
173 
174 SkinSurface*
skin_surface_create_slow(int w,int h)175 skin_surface_create_slow( int  w, int  h )
176 {
177     SDL_Surface*  surface;
178 
179     surface = _sdl_surface_create_rgb( w, h, 32, SDL_SWSURFACE );
180     if (surface == NULL) {
181         D( "could not create slow %dx%d ARGB32 surface: %s",
182             w, h, SDL_GetError() );
183         return NULL;
184     }
185     return _skin_surface_create( surface, NULL );
186 }
187 
188 
189 SkinSurface*
skin_surface_create_argb32_from(int w,int h,int pitch,uint32_t * pixels,int do_copy)190 skin_surface_create_argb32_from(
191                         int                  w,
192                         int                  h,
193                         int                  pitch,
194                         uint32_t*            pixels,
195                         int                  do_copy )
196 {
197     SDL_Surface*  surface;
198     uint32_t*     pixcopy = NULL;
199 
200     if (do_copy) {
201         size_t  size = h*pitch;
202         pixcopy = malloc( size );
203         if (pixcopy == NULL && size > 0) {
204             D( "not enough memory to create %dx%d ARGB32 surface",
205                w, h );
206             return NULL;
207         }
208         memcpy( pixcopy, pixels, size );
209     }
210 
211     surface = _sdl_surface_create_rgb_from( w, h, pitch,
212                                             pixcopy ? pixcopy : pixels,
213                                             32 );
214     if (surface == NULL) {
215         D( "could not create %dx%d slow ARGB32 surface: %s",
216             w, h, SDL_GetError() );
217         return NULL;
218     }
219     return _skin_surface_create( surface, pixcopy );
220 }
221 
222 
223 
224 
225 extern int
skin_surface_lock(SkinSurface * s,SkinSurfacePixels * pix)226 skin_surface_lock( SkinSurface*  s, SkinSurfacePixels  *pix )
227 {
228     if (!s || !s->surface) {
229         D( "error: trying to lock stale surface %p", s );
230         return -1;
231     }
232     if ( SDL_LockSurface( s->surface ) != 0 ) {
233         D( "could not lock surface %p: %s", s, SDL_GetError() );
234         return -1;
235     }
236     pix->w      = s->surface->w;
237     pix->h      = s->surface->h;
238     pix->pitch  = s->surface->pitch;
239     pix->pixels = s->surface->pixels;
240     return 0;
241 }
242 
243 /* unlock a slow surface that was previously locked */
244 extern void
skin_surface_unlock(SkinSurface * s)245 skin_surface_unlock( SkinSurface*  s )
246 {
247     if (s && s->surface)
248         SDL_UnlockSurface( s->surface );
249 }
250 
251 
252 #if 0
253 static uint32_t
254 skin_surface_map_argb( SkinSurface*  s, uint32_t  c )
255 {
256     if (s && s->surface) {
257         return SDL_MapRGBA( s->surface->format,
258                             ((c) >> 16) & 255,
259                             ((c) >> 8) & 255,
260                             ((c) & 255),
261                             ((c) >> 24) & 255 );
262     }
263     return 0x00000000;
264 }
265 #endif
266 
267 typedef struct {
268     int   x;
269     int   y;
270     int   w;
271     int   h;
272     int   sx;
273     int   sy;
274 
275     uint8_t*      dst_line;
276     int           dst_pitch;
277     SDL_Surface*  dst_lock;
278 
279     uint8_t*      src_line;
280     int           src_pitch;
281     SDL_Surface*  src_lock;
282     uint32_t      src_color;
283 
284 } SkinBlit;
285 
286 
287 static int
skin_blit_init_fill(SkinBlit * blit,SkinSurface * dst,SkinRect * dst_rect,uint32_t color)288 skin_blit_init_fill( SkinBlit*     blit,
289                      SkinSurface*  dst,
290                      SkinRect*     dst_rect,
291                      uint32_t      color )
292 {
293     int  x = dst_rect->pos.x;
294     int  y = dst_rect->pos.y;
295     int  w = dst_rect->size.w;
296     int  h = dst_rect->size.h;
297     int  delta;
298 
299     if (x < 0) {
300         w += x;
301         x  = 0;
302     }
303     delta = (x + w) - dst->surface->w;
304     if (delta > 0)
305         w -= delta;
306 
307     if (y < 0) {
308         h += y;
309         y  = 0;
310     }
311     delta = (y + h) - dst->surface->h;
312     if (delta > 0)
313         h -= delta;
314 
315     if (w <= 0 || h <= 0)
316         return 0;
317 
318     blit->x = x;
319     blit->y = y;
320     blit->w = w;
321     blit->h = h;
322 
323     if ( !SDL_LockSurface(dst->surface) )
324         return 0;
325 
326     blit->dst_lock  = dst->surface;
327     blit->dst_pitch = dst->surface->pitch;
328     blit->dst_line  = dst->surface->pixels + y*blit->dst_pitch;
329 
330     blit->src_lock  = NULL;
331     blit->src_color = color;
332 
333     return 1;
334 }
335 
336 static int
skin_blit_init_blit(SkinBlit * blit,SkinSurface * dst,SkinPos * dst_pos,SkinSurface * src,SkinRect * src_rect)337 skin_blit_init_blit( SkinBlit*     blit,
338                      SkinSurface*  dst,
339                      SkinPos*      dst_pos,
340                      SkinSurface*  src,
341                      SkinRect*     src_rect )
342 {
343     int  x  = dst_pos->x;
344     int  y  = dst_pos->y;
345     int  sx = src_rect->pos.x;
346     int  sy = src_rect->pos.y;
347     int  w  = src_rect->size.w;
348     int  h  = src_rect->size.h;
349     int  delta;
350 
351     if (x < 0) {
352         w  += x;
353         sx -= x;
354         x   = 0;
355     }
356     if (sx < 0) {
357         w  += sx;
358         x  -= sx;
359         sx  = 0;
360     }
361 
362     delta = (x + w) - dst->surface->w;
363     if (delta > 0)
364         w -= delta;
365 
366     delta = (sx + w) - src->surface->w;
367     if (delta > 0)
368         w -= delta;
369 
370     if (y < 0) {
371         h  += y;
372         sy += y;
373         y   = 0;
374     }
375     if (sy < 0) {
376         h  += sy;
377         y  -= sy;
378         sy  = 0;
379     }
380     delta = (y + h) - dst->surface->h;
381     if (delta > 0)
382         h -= delta;
383 
384     delta = (sy + h) - src->surface->h;
385 
386     if (w <= 0 || h <= 0)
387         return 0;
388 
389     blit->x = x;
390     blit->y = y;
391     blit->w = w;
392     blit->h = h;
393 
394     blit->sx = sx;
395     blit->sy = sy;
396 
397     if ( !SDL_LockSurface(dst->surface) )
398         return 0;
399 
400     blit->dst_lock  = dst->surface;
401     blit->dst_pitch = dst->surface->pitch;
402     blit->dst_line  = (uint8_t*) dst->surface->pixels + y*blit->dst_pitch;
403 
404     if ( !SDL_LockSurface(src->surface) ) {
405         SDL_UnlockSurface(dst->surface);
406         return 0;
407     }
408 
409     blit->src_lock  = src->surface;
410     blit->src_pitch = src->surface->pitch;
411     blit->src_line  = (uint8_t*) src->surface->pixels + sy*blit->src_pitch;
412 
413     return 1;
414 }
415 
416 static void
skin_blit_done(SkinBlit * blit)417 skin_blit_done( SkinBlit*  blit )
418 {
419     if (blit->src_lock)
420         SDL_UnlockSurface( blit->src_lock );
421     if (blit->dst_lock)
422         SDL_UnlockSurface( blit->dst_lock );
423     ARGB_DONE;
424 }
425 
426 typedef void (*SkinLineFillFunc)( uint32_t*  dst, uint32_t  color, int  len );
427 typedef void (*SkinLineBlitFunc)( uint32_t*  dst, const uint32_t*  src,  int  len );
428 
429 static void
skin_line_fill_copy(uint32_t * dst,uint32_t color,int len)430 skin_line_fill_copy( uint32_t*  dst, uint32_t  color, int  len )
431 {
432     uint32_t*  end = dst + len;
433 
434     while (dst + 4 <= end) {
435         dst[0] = dst[1] = dst[2] = dst[3] = color;
436         dst   += 4;
437     }
438     while (dst < end) {
439         dst[0] = color;
440         dst   += 1;
441     }
442 }
443 
444 static void
skin_line_fill_srcover(uint32_t * dst,uint32_t color,int len)445 skin_line_fill_srcover( uint32_t*  dst, uint32_t  color, int  len )
446 {
447     uint32_t*  end = dst + len;
448     uint32_t   alpha = (color >> 24);
449 
450     if (alpha == 255)
451     {
452         skin_line_fill_copy(dst, color, len);
453     }
454     else
455     {
456         ARGB_DECL(src_c);
457         ARGB_DECL_ZERO();
458 
459         alpha  = 255 - alpha;
460         alpha += (alpha >> 7);
461 
462         ARGB_UNPACK(src_c,color);
463 
464         for ( ; dst < end; dst++ )
465         {
466             ARGB_DECL(dst_c);
467 
468             ARGB_READ(dst_c,dst);
469             ARGB_MULSHIFT(dst_c,dst_c,alpha,8);
470             ARGB_ADD(dst_c,src_c);
471             ARGB_WRITE(dst_c,dst);
472         }
473     }
474 }
475 
476 static void
skin_line_fill_dstover(uint32_t * dst,uint32_t color,int len)477 skin_line_fill_dstover( uint32_t*  dst, uint32_t  color, int  len )
478 {
479     uint32_t*  end = dst + len;
480     ARGB_DECL(src_c);
481     ARGB_DECL_ZERO();
482 
483     ARGB_UNPACK(src_c,color);
484 
485     for ( ; dst < end; dst++ )
486     {
487         ARGB_DECL(dst_c);
488         ARGB_DECL(val);
489 
490         uint32_t   alpha;
491 
492         ARGB_READ(dst_c,dst);
493         alpha = 256 - (dst[0] >> 24);
494         ARGB_MULSHIFT(val,src_c,alpha,8);
495         ARGB_ADD(val,dst_c);
496         ARGB_WRITE(val,dst);
497     }
498 }
499 
500 extern void
skin_surface_fill(SkinSurface * dst,SkinRect * rect,uint32_t argb_premul,SkinBlitOp blitop)501 skin_surface_fill( SkinSurface*  dst,
502                    SkinRect*     rect,
503                    uint32_t      argb_premul,
504                    SkinBlitOp    blitop )
505 {
506     SkinLineFillFunc  fill;
507     SkinBlit          blit[1];
508 
509     switch (blitop) {
510         case SKIN_BLIT_COPY:    fill = skin_line_fill_copy; break;
511         case SKIN_BLIT_SRCOVER: fill = skin_line_fill_srcover; break;
512         case SKIN_BLIT_DSTOVER: fill = skin_line_fill_dstover; break;
513         default: return;
514     }
515 
516     if ( skin_blit_init_fill( blit, dst, rect, argb_premul ) ) {
517         uint8_t*   line  = blit->dst_line;
518         int        pitch = blit->dst_pitch;
519         uint8_t*   end   = line + pitch*blit->h;
520 
521         for ( ; line != end; line += pitch )
522             fill( (uint32_t*)line + blit->x, argb_premul, blit->w );
523     }
524 }
525 
526 
527 static void
skin_line_blit_copy(uint32_t * dst,const uint32_t * src,int len)528 skin_line_blit_copy( uint32_t*  dst, const uint32_t*  src, int  len )
529 {
530     memcpy( (char*)dst, (const char*)src, len*4 );
531 }
532 
533 
534 
535 static void
skin_line_blit_srcover(uint32_t * dst,const uint32_t * src,int len)536 skin_line_blit_srcover( uint32_t*  dst, const uint32_t*  src, int  len )
537 {
538     uint32_t*  end = dst + len;
539     ARGB_DECL_ZERO();
540 
541     for ( ; dst < end; dst++ ) {
542         ARGB_DECL(d);
543         ARGB_DECL(v);
544         uint32_t  alpha;
545 
546         alpha = (src[0] >> 24);
547         if (alpha > 0) {
548             ARGB_READ(d,dst);
549             alpha = 256 - alpha;
550             ARGB_MULSHIFT(v,d,alpha,8);
551             ARGB_ADD(v,d);
552             ARGB_WRITE(v,dst);
553         }
554     }
555 }
556 
557 static void
skin_line_blit_dstover(uint32_t * dst,const uint32_t * src,int len)558 skin_line_blit_dstover( uint32_t*  dst, const uint32_t*  src, int  len )
559 {
560     uint32_t*  end = dst + len;
561     ARGB_DECL_ZERO();
562 
563     for ( ; dst < end; dst++ ) {
564         ARGB_DECL(s);
565         ARGB_DECL(v);
566         uint32_t  alpha;
567 
568         alpha = (dst[0] >> 24);
569         if (alpha < 255) {
570             ARGB_READ(s,src);
571             alpha = 256 - alpha;
572             ARGB_MULSHIFT(v,s,alpha,8);
573             ARGB_ADD(v,s);
574             ARGB_WRITE(v,dst);
575         }
576     }
577 }
578 
579 
580 extern void
skin_surface_blit(SkinSurface * dst,SkinPos * dst_pos,SkinSurface * src,SkinRect * src_rect,SkinBlitOp blitop)581 skin_surface_blit( SkinSurface*  dst,
582                    SkinPos*      dst_pos,
583                    SkinSurface*  src,
584                    SkinRect*     src_rect,
585                    SkinBlitOp    blitop )
586 {
587     SkinLineBlitFunc  func;
588     SkinBlit          blit[1];
589 
590     switch (blitop) {
591         case SKIN_BLIT_COPY:    func = skin_line_blit_copy; break;
592         case SKIN_BLIT_SRCOVER: func = skin_line_blit_srcover; break;
593         case SKIN_BLIT_DSTOVER: func = skin_line_blit_dstover; break;
594         default: return;
595     }
596 
597     if ( skin_blit_init_blit( blit, dst, dst_pos, src, src_rect ) ) {
598         uint8_t*   line   = blit->dst_line;
599         uint8_t*   sline  = blit->src_line;
600         int        pitch  = blit->dst_pitch;
601         int        spitch = blit->src_pitch;
602         uint8_t*   end    = line + pitch*blit->h;
603 
604         for ( ; line != end; line += pitch, sline += spitch )
605             func( (uint32_t*)line + blit->x, (uint32_t*)sline + blit->sx, blit->w );
606 
607         skin_blit_done(blit);
608     }
609 }
610