rockbox/apps/recorder/resize.h
Thomas Martitz c7b8054714 Adapt the resize-on-load image scalers to support the alpha channel.
Now 32bit BMPs with alpha channel can be up- and downscaled without losing
transparency information.

git-svn-id: svn://svn.rockbox.org/rockbox/trunk@30938 a1c6a512-1295-4272-9138-f99709370657
2011-11-08 21:39:23 +00:00

177 lines
5.2 KiB
C

/***************************************************************************
* __________ __ ___.
* Open \______ \ ____ ____ | | _\_ |__ _______ ___
* Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
* Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
* Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
* \/ \/ \/ \/ \/
* $Id$
*
* Copyright (C) 2008 by Akio Idehara
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation; either version 2
* of the License, or (at your option) any later version.
*
* This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
* KIND, either express or implied.
*
****************************************************************************/
#ifndef _RESIZE_H_
#define _RESIZE_H_
#include "config.h"
#include "lcd.h"
#include "inttypes.h"
/****************************************************************
* resize_on_load()
*
* resize bitmap on load with scaling
*
* If HAVE_LCD_COLOR then this func use smooth scaling algorithm
* - downscaling both way use "Area Sampling"
* if IMG_RESIZE_BILINER or IMG_RESIZE_NEAREST is NOT set
* - otherwise "Bilinear" or "Nearest Neighbour"
*
* If !(HAVE_LCD_COLOR) then use simple scaling algorithm "Nearest Neighbour"
*
* return -1 for error
****************************************************************/
/* nothing needs the on-stack buffer right now */
#define MAX_SC_STACK_ALLOC 0
#define HAVE_UPSCALER 1
#if defined(CPU_SH)
/* perform 32x32->40 unsigned multiply, round off and return top 8 bits */
static inline uint32_t sc_mul_u32_rnd(uint32_t m, uint32_t n)
{
unsigned r, t1, t2, t3;
unsigned h = 1 << 15;
/* notation:
m = ab, n = cd
final result is (((a *c) << 32) + ((b * c + a * d) << 16) + b * d +
(1 << 31)) >> 32
*/
asm (
"swap.w %[m], %[t1]\n\t" /* t1 = ba */
"mulu %[m], %[n]\n\t" /* b * d */
"swap.w %[n], %[t3]\n\t" /* t3 = dc */
"sts macl, %[r]\n\t" /* r = b * d */
"mulu %[m], %[t3]\n\t" /* b * c */
"shlr16 %[r]\n\t"
"sts macl, %[t2]\n\t" /* t2 = b * c */
"mulu %[t1], %[t3]\n\t" /* a * c */
"add %[t2], %[r]\n\t"
"sts macl, %[t3]\n\t" /* t3 = a * c */
"mulu %[t1], %[n]\n\t" /* a * d */
"shll16 %[t3]\n\t"
"sts macl, %[t2]\n\t" /* t2 = a * d */
"add %[t2], %[r]\n\t"
"add %[t3], %[r]\n\t" /* r = ((b * d) >> 16) + (b * c + a * d) +
((a * c) << 16) */
"add %[h], %[r]\n\t" /* round result */
"shlr16 %[r]\n\t" /* truncate result */
: /* outputs */
[r] "=&r"(r),
[t1]"=&r"(t1),
[t2]"=&r"(t2),
[t3]"=&r"(t3)
: /* inputs */
[h] "r" (h),
[m] "r" (m),
[n] "r" (n)
);
return r;
}
#elif defined(TEST_SH_MATH)
static inline uint32_t sc_mul_u32_rnd(uint32_t op1, uint32_t op2)
{
uint64_t tmp = (uint64_t)op1 * op2;
tmp += 1LU << 31;
tmp >>= 32;
return tmp;
}
#else
#define SC_OUT(n, c) (((n) + (1 << 23)) >> 24)
#endif
#ifndef SC_OUT
#define SC_OUT(n, c) (sc_mul_u32_rnd(n, (c)->recip))
#endif
struct img_part {
int len;
#if !defined(HAVE_LCD_COLOR)
uint8_t *buf;
#else
struct uint8_rgb* buf;
#endif
};
#ifdef HAVE_LCD_COLOR
/* intermediate type used by the scaler for color output. greyscale version
uses uint32_t
*/
struct uint32_argb {
uint32_t r;
uint32_t g;
uint32_t b;
uint32_t a;
};
#endif
/* struct which contains various parameters shared between vertical scaler,
horizontal scaler, and row output
*/
struct scaler_context {
#if defined(CPU_SH) || defined(TEST_SH_MATH)
uint32_t recip;
#else
uint32_t h_i_val;
uint32_t h_o_val;
uint32_t v_i_val;
uint32_t v_o_val;
#endif
struct bitmap *bm;
struct dim *src;
unsigned char *buf;
bool dither;
int len;
void *args;
struct img_part* (*store_part)(void *);
void (*output_row)(uint32_t,void*,struct scaler_context*);
bool (*h_scaler)(void*,struct scaler_context*, bool);
};
#if defined(HAVE_LCD_COLOR)
#define IF_PIX_FMT(...) __VA_ARGS__
#else
#define IF_PIX_FMT(...)
#endif
struct custom_format {
void (*output_row_8)(uint32_t,void*, struct scaler_context*);
#if defined(HAVE_LCD_COLOR)
void (*output_row_32[2])(uint32_t,void*, struct scaler_context*);
#else
void (*output_row_32)(uint32_t,void*, struct scaler_context*);
#endif
unsigned int (*get_size)(struct bitmap *bm);
};
struct rowset;
extern const struct custom_format format_native;
int recalc_dimension(struct dim *dst, struct dim *src);
int resize_on_load(struct bitmap *bm, bool dither,
struct dim *src, struct rowset *tmp_row,
unsigned char *buf, unsigned int len,
const struct custom_format *cformat,
IF_PIX_FMT(int format_index,)
struct img_part* (*store_part)(void *args),
void *args);
#endif /* _RESIZE_H_ */