231 lines
4.3 KiB
C
231 lines
4.3 KiB
C
#include <u.h>
|
|
#include <libc.h>
|
|
#include <bio.h>
|
|
#include "sky.h"
|
|
|
|
static void unshuffle(Pix*, int, int, Pix*);
|
|
static void unshuffle1(Pix*, int, Pix*);
|
|
|
|
void
|
|
hinv(Pix *a, int nx, int ny)
|
|
{
|
|
int nmax, log2n, h0, hx, hy, hc, i, j, k;
|
|
int nxtop, nytop, nxf, nyf, c;
|
|
int oddx, oddy;
|
|
int shift;
|
|
int s10, s00;
|
|
Pix *tmp;
|
|
|
|
/*
|
|
* log2n is log2 of max(nx, ny) rounded up to next power of 2
|
|
*/
|
|
nmax = ny;
|
|
if(nx > nmax)
|
|
nmax = nx;
|
|
log2n = log(nmax)/LN2 + 0.5;
|
|
if(nmax > (1<<log2n))
|
|
log2n++;
|
|
|
|
/*
|
|
* get temporary storage for shuffling elements
|
|
*/
|
|
tmp = (Pix*)malloc(((nmax+1)/2) * sizeof(*tmp));
|
|
if(tmp == nil) {
|
|
fprint(2, "hinv: insufficient memory\n");
|
|
exits("memory");
|
|
}
|
|
|
|
/*
|
|
* do log2n expansions
|
|
*
|
|
* We're indexing a as a 2-D array with dimensions (nx,ny).
|
|
*/
|
|
shift = 1;
|
|
nxtop = 1;
|
|
nytop = 1;
|
|
nxf = nx;
|
|
nyf = ny;
|
|
c = 1<<log2n;
|
|
for(k = log2n-1; k>=0; k--) {
|
|
/*
|
|
* this somewhat cryptic code generates the sequence
|
|
* ntop[k-1] = (ntop[k]+1)/2, where ntop[log2n] = n
|
|
*/
|
|
c = c>>1;
|
|
nxtop = nxtop<<1;
|
|
nytop = nytop<<1;
|
|
if(nxf <= c)
|
|
nxtop--;
|
|
else
|
|
nxf -= c;
|
|
if(nyf <= c)
|
|
nytop--;
|
|
else
|
|
nyf -= c;
|
|
|
|
/*
|
|
* halve divisors on last pass
|
|
*/
|
|
if(k == 0)
|
|
shift = 0;
|
|
|
|
/*
|
|
* unshuffle in each dimension to interleave coefficients
|
|
*/
|
|
for(i = 0; i<nxtop; i++)
|
|
unshuffle1(&a[ny*i], nytop, tmp);
|
|
for(j = 0; j<nytop; j++)
|
|
unshuffle(&a[j], nxtop, ny, tmp);
|
|
oddx = nxtop & 1;
|
|
oddy = nytop & 1;
|
|
for(i = 0; i<nxtop-oddx; i += 2) {
|
|
s00 = ny*i; /* s00 is index of a[i,j] */
|
|
s10 = s00+ny; /* s10 is index of a[i+1,j] */
|
|
for(j = 0; j<nytop-oddy; j += 2) {
|
|
/*
|
|
* Multiply h0,hx,hy,hc by 2 (1 the last time through).
|
|
*/
|
|
h0 = a[s00 ] << shift;
|
|
hx = a[s10 ] << shift;
|
|
hy = a[s00+1] << shift;
|
|
hc = a[s10+1] << shift;
|
|
|
|
/*
|
|
* Divide sums by 4 (shift right 2 bits).
|
|
* Add 1 to round -- note that these values are always
|
|
* positive so we don't need to do anything special
|
|
* for rounding negative numbers.
|
|
*/
|
|
a[s10+1] = (h0 + hx + hy + hc + 2) >> 2;
|
|
a[s10 ] = (h0 + hx - hy - hc + 2) >> 2;
|
|
a[s00+1] = (h0 - hx + hy - hc + 2) >> 2;
|
|
a[s00 ] = (h0 - hx - hy + hc + 2) >> 2;
|
|
s00 += 2;
|
|
s10 += 2;
|
|
}
|
|
if(oddy) {
|
|
/*
|
|
* do last element in row if row length is odd
|
|
* s00+1, s10+1 are off edge
|
|
*/
|
|
h0 = a[s00 ] << shift;
|
|
hx = a[s10 ] << shift;
|
|
a[s10 ] = (h0 + hx + 2) >> 2;
|
|
a[s00 ] = (h0 - hx + 2) >> 2;
|
|
}
|
|
}
|
|
if(oddx) {
|
|
/*
|
|
* do last row if column length is odd
|
|
* s10, s10+1 are off edge
|
|
*/
|
|
s00 = ny*i;
|
|
for(j = 0; j<nytop-oddy; j += 2) {
|
|
h0 = a[s00 ] << shift;
|
|
hy = a[s00+1] << shift;
|
|
a[s00+1] = (h0 + hy + 2) >> 2;
|
|
a[s00 ] = (h0 - hy + 2) >> 2;
|
|
s00 += 2;
|
|
}
|
|
if(oddy) {
|
|
/*
|
|
* do corner element if both row and column lengths are odd
|
|
* s00+1, s10, s10+1 are off edge
|
|
*/
|
|
h0 = a[s00 ] << shift;
|
|
a[s00 ] = (h0 + 2) >> 2;
|
|
}
|
|
}
|
|
}
|
|
free(tmp);
|
|
}
|
|
|
|
static
|
|
void
|
|
unshuffle(Pix *a, int n, int n2, Pix *tmp)
|
|
{
|
|
int i;
|
|
int nhalf, twon2, n2xnhalf;
|
|
Pix *p1, *p2, *pt;
|
|
|
|
twon2 = n2<<1;
|
|
nhalf = (n+1)>>1;
|
|
n2xnhalf = n2*nhalf; /* pointer to a[i] */
|
|
|
|
/*
|
|
* copy 2nd half of array to tmp
|
|
*/
|
|
pt = tmp;
|
|
p1 = &a[n2xnhalf]; /* pointer to a[i] */
|
|
for(i=nhalf; i<n; i++) {
|
|
*pt = *p1;
|
|
pt++;
|
|
p1 += n2;
|
|
}
|
|
|
|
/*
|
|
* distribute 1st half of array to even elements
|
|
*/
|
|
p2 = &a[n2xnhalf]; /* pointer to a[i] */
|
|
p1 = &a[n2xnhalf<<1]; /* pointer to a[2*i] */
|
|
for(i=nhalf-1; i>=0; i--) {
|
|
p1 -= twon2;
|
|
p2 -= n2;
|
|
*p1 = *p2;
|
|
}
|
|
|
|
/*
|
|
* now distribute 2nd half of array (in tmp) to odd elements
|
|
*/
|
|
pt = tmp;
|
|
p1 = &a[n2]; /* pointer to a[i] */
|
|
for(i=1; i<n; i+=2) {
|
|
*p1 = *pt;
|
|
p1 += twon2;
|
|
pt++;
|
|
}
|
|
}
|
|
|
|
static
|
|
void
|
|
unshuffle1(Pix *a, int n, Pix *tmp)
|
|
{
|
|
int i;
|
|
int nhalf;
|
|
Pix *p1, *p2, *pt;
|
|
|
|
nhalf = (n+1) >> 1;
|
|
|
|
/*
|
|
* copy 2nd half of array to tmp
|
|
*/
|
|
pt = tmp;
|
|
p1 = &a[nhalf]; /* pointer to a[i] */
|
|
for(i=nhalf; i<n; i++) {
|
|
*pt = *p1;
|
|
pt++;
|
|
p1++;
|
|
}
|
|
|
|
/*
|
|
* distribute 1st half of array to even elements
|
|
*/
|
|
p2 = &a[nhalf]; /* pointer to a[i] */
|
|
p1 = &a[nhalf<<1]; /* pointer to a[2*i] */
|
|
for(i=nhalf-1; i>=0; i--) {
|
|
p1 -= 2;
|
|
p2--;
|
|
*p1 = *p2;
|
|
}
|
|
|
|
/*
|
|
* now distribute 2nd half of array (in tmp) to odd elements
|
|
*/
|
|
pt = tmp;
|
|
p1 = &a[1]; /* pointer to a[i] */
|
|
for(i=1; i<n; i+=2) {
|
|
*p1 = *pt;
|
|
p1 += 2;
|
|
pt++;
|
|
}
|
|
}
|