/** * * ***** BEGIN GPL/BL DUAL LICENSE BLOCK ***** * * 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. The Blender * Foundation also sells licenses for use in proprietary software under * the Blender License. See http://www.blender.org/BL/ for information * about this. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software Foundation, * Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. * * The Original Code is Copyright (C) 2001-2002 by NaN Holding BV. * All rights reserved. * * The Original Code is: all of this file. * * Contributor(s): none yet. * * ***** END GPL/BL DUAL LICENSE BLOCK ***** * allocimbuf.c * * $Id$ */ #include "BLI_blenlib.h" #include "imbuf.h" #include "imbuf_patch.h" #include "IMB_imbuf_types.h" #include "IMB_imbuf.h" #include "IMB_allocimbuf.h" #include "IMB_filter.h" /************************************************************************/ /* SCALING */ /************************************************************************/ struct ImBuf *IMB_half_x(struct ImBuf *ibuf1) { struct ImBuf *ibuf2; uchar *p1,*_p1,*dest; short a,r,g,b,x,y; float af,rf,gf,bf, *p1f, *_p1f, *destf; int do_float = 0; if (ibuf1==NULL) return (0); if (ibuf1->rect==NULL) return (0); if (ibuf1->rect_float) do_float = 1; if (ibuf1->x <= 1) return(IMB_dupImBuf(ibuf1)); ibuf2 = IMB_allocImBuf((ibuf1->x)/2, ibuf1->y, ibuf1->depth, ibuf1->flags, 0); if (ibuf2==NULL) return (0); _p1 = (uchar *) ibuf1->rect; dest=(uchar *) ibuf2->rect; _p1f = ibuf1->rect_float; destf= ibuf2->rect_float; for(y=ibuf2->y;y>0;y--){ p1 = _p1; p1f = _p1f; for(x = ibuf2->x ; x>0 ; x--){ a = *(p1++) ; b = *(p1++) ; g = *(p1++) ; r = *(p1++); a += *(p1++) ; b += *(p1++) ; g += *(p1++) ; r += *(p1++); *(dest++) = a >> 1; *(dest++) = b >> 1; *(dest++) = g >> 1; *(dest++) = r >> 1; if (do_float) { af = *(p1f++); bf = *(p1f++); gf = *(p1f++); rf = *(p1f++); af += *(p1f++); bf += *(p1f++); gf += *(p1f++); rf += *(p1f++); *(destf++) = 0.5f*af; *(destf++) = 0.5f*bf; *(destf++) = 0.5f*gf; *(destf++) = 0.5f*rf; } } _p1 += (ibuf1->x << 2); if (do_float) _p1f += (ibuf1->x << 2); } return (ibuf2); } struct ImBuf *IMB_double_fast_x(struct ImBuf *ibuf1) { struct ImBuf *ibuf2; int *p1,*dest, i, col, do_float=0; float *p1f, *destf, colf; if (ibuf1==NULL) return (0); if (ibuf1->rect==NULL) return (0); if (ibuf1->rect_float) do_float = 1; ibuf2 = IMB_allocImBuf(2 * ibuf1->x , ibuf1->y , ibuf1->depth, ibuf1->flags, 0); if (ibuf2==NULL) return (0); p1 = (int *) ibuf1->rect; dest=(int *) ibuf2->rect; p1f = ibuf1->rect_float; destf = ibuf2->rect_float; for(i = ibuf1->y * ibuf1->x ; i>0 ; i--) { col = *p1++; *dest++ = col; *dest++ = col; if (do_float) { colf = *p1f++; *destf++ = col; *destf++ = col; } } return (ibuf2); } struct ImBuf *IMB_double_x(struct ImBuf *ibuf1) { struct ImBuf *ibuf2; if (ibuf1==NULL) return (0); if (ibuf1->rect==NULL) return (0); ibuf2 = IMB_double_fast_x(ibuf1); imb_filterx(ibuf2); return (ibuf2); } struct ImBuf *IMB_half_y(struct ImBuf *ibuf1) { struct ImBuf *ibuf2; uchar *p1,*p2,*_p1,*dest; short a,r,g,b,x,y; int do_float = 0; float af,rf,gf,bf,*p1f,*p2f,*_p1f,*destf; p1f = NULL; p2f = NULL; if (ibuf1==NULL) return (0); if (ibuf1->rect==NULL) return (0); if (ibuf1->rect_float) do_float = 1; if (ibuf1->y <= 1) return(IMB_dupImBuf(ibuf1)); ibuf2 = IMB_allocImBuf(ibuf1->x , (ibuf1->y) / 2 , ibuf1->depth, ibuf1->flags, 0); if (ibuf2==NULL) return (0); _p1 = (uchar *) ibuf1->rect; dest=(uchar *) ibuf2->rect; _p1f = ibuf1->rect_float; destf= ibuf2->rect_float; for(y=ibuf2->y ; y>0 ; y--){ p1 = _p1; p2 = _p1 + (ibuf1->x << 2); if (do_float) { p1f = _p1f; p2f = _p1f + (ibuf1->x << 2); } for(x = ibuf2->x ; x>0 ; x--){ a = *(p1++) ; b = *(p1++) ; g = *(p1++) ; r = *(p1++); a += *(p2++) ; b += *(p2++) ; g += *(p2++) ; r += *(p2++); *(dest++) = a >> 1; *(dest++) = b >> 1; *(dest++) = g >> 1; *(dest++) = r >> 1; if (do_float) { af = *(p1f++) ; bf = *(p1f++) ; gf = *(p1f++) ; rf = *(p1f++); af += *(p2f++) ; bf += *(p2f++) ; gf += *(p2f++) ; rf += *(p2f++); *(destf++) = 0.5f*af; *(destf++) = 0.5f*bf; *(destf++) = 0.5f*gf; *(destf++) = 0.5f*rf; } } _p1 += (ibuf1->x << 3); if (do_float) _p1f += (ibuf1->x << 3); } return (ibuf2); } struct ImBuf *IMB_double_fast_y(struct ImBuf *ibuf1) { struct ImBuf *ibuf2; int *p1, *dest1, *dest2; float *p1f, *dest1f, *dest2f; short x,y; int do_float =0; if (ibuf1==NULL) return (0); if (ibuf1->rect==NULL) return (0); if (ibuf1->rect_float) do_float =1; ibuf2 = IMB_allocImBuf(ibuf1->x , 2 * ibuf1->y , ibuf1->depth, ibuf1->flags, 0); if (ibuf2==NULL) return (0); p1 = (int *) ibuf1->rect; dest1=(int *) ibuf2->rect; p1f = ibuf1->rect_float; dest1f= ibuf2->rect_float; for(y = ibuf1->y ; y>0 ; y--){ dest2 = dest1 + ibuf2->x; for(x = ibuf2->x ; x>0 ; x--) *dest1++ = *dest2++ = *p1++; dest1 = dest2; if (do_float) { dest2f = dest1f + ibuf2->x; for(x = ibuf2->x ; x>0 ; x--) *dest1f++ = *dest2f++ = *p1f++; dest1f = dest2f; } } return (ibuf2); } struct ImBuf *IMB_double_y(struct ImBuf *ibuf1) { struct ImBuf *ibuf2; if (ibuf1==NULL) return (0); if (ibuf1->rect==NULL) return (0); ibuf2 = IMB_double_fast_y(ibuf1); IMB_filtery(ibuf2); return (ibuf2); } struct ImBuf *IMB_onehalf(struct ImBuf *ibuf1) { struct ImBuf *ibuf2; uchar *p1, *p2, *dest; float *p1f, *destf, *p2f = NULL; int x,y; int do_float =0; if (ibuf1==NULL) return (0); if (ibuf1->rect==NULL) return (0); if (ibuf1->rect_float) do_float = 1; if (ibuf1->x <= 1) return(IMB_half_y(ibuf1)); if (ibuf1->y <= 1) return(IMB_half_x(ibuf1)); ibuf2=IMB_allocImBuf((ibuf1->x)/2, (ibuf1->y)/2, ibuf1->depth, ibuf1->flags, 0); if (ibuf2==NULL) return (0); p1f = ibuf1->rect_float; destf=ibuf2->rect_float; p1 = (uchar *) ibuf1->rect; dest=(uchar *) ibuf2->rect; for(y=ibuf2->y;y>0;y--){ p2 = p1 + (ibuf1->x << 2); if (do_float) p2f = p1f + (ibuf1->x << 2); for(x=ibuf2->x;x>0;x--){ dest[0] = (p1[0] + p2[0] + p1[4] + p2[4]) >> 2; dest[1] = (p1[1] + p2[1] + p1[5] + p2[5]) >> 2; dest[2] = (p1[2] + p2[2] + p1[6] + p2[6]) >> 2; dest[3] = (p1[3] + p2[3] + p1[7] + p2[7]) >> 2; p1 += 8; p2 += 8; dest += 4; if (do_float){ destf[0] = 0.25f*(p1f[0] + p2f[0] + p1f[4] + p2f[4]); destf[1] = 0.25f*(p1f[1] + p2f[1] + p1f[5] + p2f[5]); destf[2] = 0.25f*(p1f[2] + p2f[2] + p1f[6] + p2f[6]); destf[3] = 0.25f*(p1f[3] + p2f[3] + p1f[7] + p2f[7]); p1f += 8; p2f += 8; destf += 4; } } p1=p2; if (do_float) p1f=p2f; if(ibuf1->x & 1) { p1+=4; if (do_float) p1f+=4; } } return (ibuf2); } struct ImBuf *IMB_onethird(struct ImBuf *ibuf1) { struct ImBuf *ibuf2; uchar *p1,*p2,*p3,*dest; float *p1f, *p2f, *p3f, *destf; int do_float=0; short a,r,g,b,x,y,i; float af,rf,gf,bf; p2f = NULL; p3f = NULL; if (ibuf1==NULL) return (0); if (ibuf1->rect==NULL) return (0); if (ibuf1->rect_float) do_float = 1; ibuf2=IMB_allocImBuf((ibuf1->x)/3, (ibuf1->y)/3, ibuf1->depth, ibuf1->flags, 0); if (ibuf2==NULL) return (0); p1f = ibuf1->rect_float; destf = ibuf2->rect_float; p1 = (uchar *) ibuf1->rect; dest=(uchar *) ibuf2->rect; for(y=ibuf2->y;y>0;y--){ p2 = p1 + (ibuf1->x << 2); p3 = p2 + (ibuf1->x << 2); if (do_float) { p2f = p1f + (ibuf1->x <<2); p3f = p2f + (ibuf1->x <<2); } for(x=ibuf2->x;x>0;x--){ a=r=g=b=0; af=rf=gf=bf=0; for (i=3;i>0;i--){ a += *(p1++) + *(p2++) + *(p3++); b += *(p1++) + *(p2++) + *(p3++); g += *(p1++) + *(p2++) + *(p3++); r += *(p1++) + *(p2++) + *(p3++); if (do_float) { af += *(p1f++) + *(p2f++) + *(p3f++); bf += *(p1f++) + *(p2f++) + *(p3f++); gf += *(p1f++) + *(p2f++) + *(p3f++); rf += *(p1f++) + *(p2f++) + *(p3f++); } } *(dest++) = a/9; *(dest++) = b/9; *(dest++) = g/9; *(dest++) = r/9; if (do_float) { *(destf++) = af/9.0f; *(destf++) = bf/9.0f; *(destf++) = gf/9.0f; *(destf++) = rf/9.0f; } } p1=p3; if (do_float) p1f = p3f; } return (ibuf2); } struct ImBuf *IMB_halflace(struct ImBuf *ibuf1) { struct ImBuf *ibuf2; uchar *p1,*p2,*dest; float *p1f,*p2f,*destf; short a,r,g,b,x,y,i; float af,rf,gf,bf; int do_float = 0; p2f = NULL; if (ibuf1==NULL) return (0); if (ibuf1->rect==NULL) return (0); if (ibuf1->rect_float) do_float = 1; ibuf2=IMB_allocImBuf((ibuf1->x)/4, (ibuf1->y)/2, ibuf1->depth, ibuf1->flags, 0); if (ibuf2==NULL) return (0); p1f = ibuf1->rect_float; destf= ibuf2->rect_float; p1 = (uchar *) ibuf1->rect; dest=(uchar *) ibuf2->rect; for(y= ibuf2->y / 2 ; y>0;y--){ p2 = p1 + (ibuf1->x << 3); if (do_float) p2f = p1f + (ibuf1->x << 3); for(x = 2 * ibuf2->x;x>0;x--){ a=r=g=b=0; af=rf=gf=bf=0; for (i=4;i>0;i--){ a += *(p1++) + *(p2++); b += *(p1++) + *(p2++); g += *(p1++) + *(p2++); r += *(p1++) + *(p2++); if (do_float) { af += *(p1f++) + *(p2f++); bf += *(p1f++) + *(p2f++); gf += *(p1f++) + *(p2f++); rf += *(p1f++) + *(p2f++); } } *(dest++) = a >> 3; *(dest++) = b >> 3; *(dest++) = g >> 3; *(dest++) = r >> 3; if (do_float) { *(destf++) = 0.125f*af; *(destf++) = 0.125f*bf; *(destf++) = 0.125f*gf; *(destf++) = 0.125f*rf; } } p1 = p2; if (do_float) p1f = p2f; } return (ibuf2); } static struct ImBuf *scaledownx(struct ImBuf *ibuf, int newx) { uchar *rect,*_newrect,*newrect; float *rectf,*_newrectf,*newrectf; float sample, add, val, nval, valf, nvalf; int x, y, i, do_float=0; rectf = NULL; _newrectf= NULL; newrectf = NULL; nval = 0; nvalf = 0; if (ibuf==NULL) return(0); if (ibuf->rect==NULL) return(ibuf); if (ibuf->rect_float) { do_float = 1; _newrectf = MEM_mallocN(newx * ibuf->y * sizeof(float) * 4, "scaledownxf"); if (_newrectf==NULL) return(ibuf); } _newrect = MEM_mallocN(newx * ibuf->y * sizeof(int), "scaledownx"); if (_newrect==NULL) return(ibuf); add = (ibuf->x - 0.001) / newx; /* all four components, rgba/abgr */ for(i=3 ; i >= 0 ; i--){ rect = (uchar *) ibuf->rect; rect += i; newrect = _newrect + i; if (do_float) { rectf = ibuf->rect_float; rectf += i; newrectf = _newrectf + i; } for (y = ibuf->y; y>0 ; y--){ val = sample = valf = 0.0; for (x = newx ; x>0 ; x--){ nval = - val * sample; if (do_float) nvalf = - valf * sample; sample += add; while (sample >= 1.0){ sample -= 1.0; nval += *rect; rect += 4; if (do_float) { nvalf += *rectf; rectf += 4; } } val = *rect; rect += 4; nval += sample * val; if (do_float) { valf = *rectf; rectf += 4; nvalf += sample * valf; *newrectf = (nvalf/add) + 0.5; newrectf += 4; } sample -= 1.0; *newrect = (nval/add) + 0.5; newrect += 4; } } } imb_freerectImBuf(ibuf); ibuf->mall |= IB_rect; ibuf->rect = (unsigned int *) _newrect; if (do_float) { imb_freerectfloatImBuf(ibuf); ibuf->mall |= IB_rectfloat; ibuf->rect_float = _newrectf; } ibuf->x = newx; return(ibuf); } static struct ImBuf *scaledowny(struct ImBuf *ibuf, int newy) { uchar *rect, *_newrect, *newrect; float *rectf, *_newrectf, *newrectf; float sample, add, val, nval, valf, nvalf; int x, y, i, skipx, do_float = 0; rectf= NULL; _newrectf = NULL; newrectf = NULL; nval = 0; nvalf = 0; if (ibuf==NULL) return(0); if (ibuf->rect==NULL) return(ibuf); if (ibuf->rect_float) { do_float = 1; _newrectf = MEM_mallocN(newy * ibuf->x * sizeof(float) * 4, "scaldownyf"); if (_newrectf==NULL) return(ibuf); } _newrect = MEM_mallocN(newy * ibuf->x * sizeof(int), "scaledowny"); if (_newrect==NULL) return(ibuf); add = (ibuf->y - 0.001) / newy; skipx = 4 * ibuf->x; /* all four components, rgba/abgr */ for(i=3 ; i>=0 ; i--){ for (x = skipx - 4; x>=0 ; x-= 4){ rect = ((uchar *) ibuf->rect) + i + x; newrect = _newrect + i + x; if (do_float) { rectf = ((float *) ibuf->rect_float) + i + x; newrectf = _newrectf + i + x; } val = sample = 0.0; for (y = newy ; y>0 ; y--){ nval = - val * sample; if (do_float) nvalf = - val * sample; sample += add; while (sample >= 1.0){ sample -= 1.0; nval += *rect; rect += skipx; if (do_float) { nvalf += *rectf; rectf += skipx; } } val = *rect; rect += skipx; nval += sample * val; *newrect = (nval/add) + 0.5; newrect += skipx; if (do_float) { valf = *rectf; rectf += skipx; nvalf += sample * valf; *newrectf = (nvalf/add) + 0.5; newrectf += skipx; } sample -= 1.0; } } } imb_freerectImBuf(ibuf); ibuf->mall |= IB_rect; ibuf->rect = (unsigned int *) _newrect; if (do_float) { imb_freerectfloatImBuf(ibuf); ibuf->mall |= IB_rectfloat; ibuf->rect_float = (float *) _newrectf; } ibuf->y = newy; return(ibuf); } static struct ImBuf *scaleupx(struct ImBuf *ibuf, int newx) { uchar *rect,*_newrect,*newrect; float *rectf,*_newrectf,*newrectf; float sample,add; float val_a,nval_a,diff_a; float val_b,nval_b,diff_b; float val_g,nval_g,diff_g; float val_r,nval_r,diff_r; float val_af,nval_af,diff_af; float val_bf,nval_bf,diff_bf; float val_gf,nval_gf,diff_gf; float val_rf,nval_rf,diff_rf; int x,y, do_float = 0; val_af = nval_af = diff_af = val_bf = nval_bf = diff_bf = 0; val_gf = nval_gf = diff_gf = val_rf = nval_rf = diff_rf = 0; if (ibuf==NULL) return(0); if (ibuf->rect==NULL) return(ibuf); _newrect = MEM_mallocN(newx * ibuf->y * sizeof(int), "scaleupx"); if (_newrect==NULL) return(ibuf); if (ibuf->rect_float) { do_float = 1; _newrectf = MEM_mallocN(newx * ibuf->y * sizeof(float) * 4, "scaleupxf"); if (_newrectf==NULL) return(ibuf); } add = (ibuf->x - 1.001) / (newx - 1.0); rect = (uchar *) ibuf->rect; rectf = (float *) ibuf->rect_float; newrect = _newrect; newrectf = _newrectf = NULL; for (y = ibuf->y; y>0 ; y--){ sample = 0; val_a = rect[0] ; nval_a = rect[4]; diff_a = nval_a - val_a ; val_a += 0.5; val_b = rect[1] ; nval_b = rect[5]; diff_b = nval_b - val_b ; val_b += 0.5; val_g = rect[2] ; nval_g = rect[6]; diff_g = nval_g - val_g ; val_g += 0.5; val_r = rect[3] ; nval_r = rect[7]; diff_r = nval_r - val_r ; val_r += 0.5; rect += 8; if (do_float) { val_af = rectf[0] ; nval_af = rectf[4]; diff_af = nval_af - val_af; val_af += 0.5; val_bf = rectf[1] ; nval_bf = rectf[5]; diff_bf = nval_bf - val_bf; val_bf += 0.5; val_gf = rectf[2] ; nval_gf = rectf[6]; diff_gf = nval_gf - val_gf; val_gf += 0.5; val_rf = rectf[3] ; nval_rf = rectf[7]; diff_rf = nval_rf - val_rf; val_rf += 0.5; rectf += 8; } for (x = newx ; x>0 ; x--){ if (sample >= 1.0){ sample -= 1.0; val_a = nval_a ; nval_a = rect[0] ; diff_a = nval_a - val_a ; val_a += 0.5; val_b = nval_b ; nval_b = rect[1] ; diff_b = nval_b - val_b ; val_b += 0.5; val_g = nval_g ; nval_g = rect[2] ; diff_g = nval_g - val_g ; val_g += 0.5; val_r = nval_r ; nval_r = rect[3] ; diff_r = nval_r - val_r ; val_r += 0.5; rect += 4; if (do_float) { val_af = nval_af ; nval_af = rectf[0] ; diff_af = nval_af - val_af ; val_af += 0.5; val_bf = nval_bf ; nval_bf = rectf[1] ; diff_bf = nval_bf - val_bf ; val_bf += 0.5; val_gf = nval_gf ; nval_gf = rectf[2] ; diff_gf = nval_gf - val_gf ; val_gf += 0.5; val_rf = nval_rf ; nval_rf = rectf[3] ; diff_rf = nval_rf - val_rf; val_rf += 0.5; rectf += 4; } } newrect[0] = val_a + sample * diff_a; newrect[1] = val_b + sample * diff_b; newrect[2] = val_g + sample * diff_g; newrect[3] = val_r + sample * diff_r; newrect += 4; if (do_float) { newrectf[0] = val_af + sample * diff_af; newrectf[1] = val_bf + sample * diff_bf; newrectf[2] = val_gf + sample * diff_gf; newrectf[3] = val_rf + sample * diff_rf; newrectf += 4; } sample += add; } } imb_freerectImBuf(ibuf); ibuf->mall |= IB_rect; ibuf->rect = (unsigned int *) _newrect; if (do_float) { imb_freerectfloatImBuf(ibuf); ibuf->mall |= IB_rectfloat; ibuf->rect_float = (float *) _newrectf; } ibuf->x = newx; return(ibuf); } static struct ImBuf *scaleupy(struct ImBuf *ibuf, int newy) { uchar *rect,*_newrect,*newrect; float *rectf = NULL, *newrectf = NULL, *_newrectf = NULL; float sample,add,val,nval,diff, valf=0.0f, nvalf=0.0f, difff=0.0f; int x,y,i,skipx, do_float = 0; if (ibuf==NULL) return(0); if (ibuf->rect==NULL) return(ibuf); _newrect = MEM_mallocN(newy * ibuf->x * sizeof(int), "scaleupy"); if (_newrect==NULL) return(ibuf); if (ibuf->rect_float) { do_float = 1; _newrectf = MEM_mallocN(newy * ibuf->y * sizeof(float) * 4, "scaleupyf"); if (_newrectf==NULL) return(ibuf); } add = (ibuf->y - 1.001) / (newy - 1.0); skipx = 4 * ibuf->x; /* all four components, rgba/abgr */ for(i=3 ; i>=0 ; i--){ for (x = skipx - 4; x >= 0 ; x -= 4){ rect = (uchar *) ibuf->rect; rect += i + x; newrect = _newrect + i + x; if (do_float) { rectf = ibuf->rect_float; rectf += i * x; newrectf = _newrectf + i + x; valf = *rectf; rectf += skipx; nvalf = *rectf; rectf += skipx; difff = nvalf - valf; valf += 0.5; } sample = 0; val = *rect ; rect += skipx; nval = *rect; rect += skipx; diff = nval - val; val += 0.5; for (y = newy ; y>0 ; y--){ if (sample >= 1.0){ sample -= 1.0; val = nval; nval = *rect; rect += skipx; diff = nval - val; val += 0.5; if (do_float) { valf = nvalf; nvalf = *rectf; rectf += skipx; difff = nvalf - valf; valf += 0.5; } } *newrect = val + sample * diff; newrect += skipx; if (do_float) { *newrectf = valf + sample * difff; newrectf += skipx; } sample += add; } } } imb_freerectImBuf(ibuf); ibuf->mall |= IB_rect; ibuf->rect = (unsigned int *) _newrect; if(do_float) { imb_freerectfloatImBuf(ibuf); ibuf->mall |= IB_rectfloat; ibuf->rect_float = _newrectf; } ibuf->y = newy; return(ibuf); } /* no float buf needed here! */ static void scalefast_Z_ImBuf(ImBuf *ibuf, short newx, short newy) { unsigned int *rect, *_newrect, *newrect; int x, y; int ofsx, ofsy, stepx, stepy; if (ibuf->zbuf) { _newrect = MEM_mallocN(newx * newy * sizeof(int), "z rect"); if (_newrect==NULL) return; stepx = (65536.0 * (ibuf->x - 1.0) / (newx - 1.0)) + 0.5; stepy = (65536.0 * (ibuf->y - 1.0) / (newy - 1.0)) + 0.5; ofsy = 32768; newrect = _newrect; for (y = newy; y > 0 ; y--){ rect = (unsigned int*) ibuf->zbuf; rect += (ofsy >> 16) * ibuf->x; ofsy += stepy; ofsx = 32768; for (x = newx ; x > 0 ; x--){ *newrect++ = rect[ofsx >> 16]; ofsx += stepx; } } IMB_freezbufImBuf(ibuf); ibuf->mall |= IB_zbuf; ibuf->zbuf = (int*) _newrect; } } struct ImBuf *IMB_scaleImBuf(struct ImBuf * ibuf, short newx, short newy) { if (ibuf==NULL) return (0); if (ibuf->rect==NULL) return (ibuf); // scaleup / scaledown functions below change ibuf->x and ibuf->y // so we first scale the Z-buffer (if any) scalefast_Z_ImBuf(ibuf, newx, newy); if (newx < ibuf->x) if (newx) scaledownx(ibuf,newx); if (newy < ibuf->y) if (newy) scaledowny(ibuf,newy); if (newx > ibuf->x) if (newx) scaleupx(ibuf,newx); if (newy > ibuf->y) if (newy) scaleupy(ibuf,newy); return(ibuf); } struct imbufRGBA { float r, g, b, a; }; struct ImBuf *IMB_scalefastImBuf(struct ImBuf *ibuf, short newx, short newy) { unsigned int *rect,*_newrect,*newrect; struct imbufRGBA *rectf, *_newrectf, *newrectf; int x,y, do_float=0, do_rect=0; int ofsx,ofsy,stepx,stepy; rect = NULL; _newrect = NULL; newrect = NULL; rectf = NULL; _newrectf = NULL; newrectf = NULL; if (ibuf==NULL) return(0); if (ibuf->rect) do_rect = 1; if (ibuf->rect_float) do_float = 1; if (do_rect==0 && do_float==0) return(ibuf); if (newx == ibuf->x && newy == ibuf->y) return(ibuf); if(do_rect) { _newrect = MEM_mallocN(newx * newy * sizeof(int), "scalefastimbuf"); if (_newrect==NULL) return(ibuf); newrect = _newrect; } if (do_float) { _newrectf = MEM_mallocN(newx * newy * sizeof(float) * 4, "scalefastimbuf f"); if (_newrectf==NULL) return(ibuf); newrectf = _newrectf; } stepx = (65536.0 * (ibuf->x - 1.0) / (newx - 1.0)) + 0.5; stepy = (65536.0 * (ibuf->y - 1.0) / (newy - 1.0)) + 0.5; ofsy = 32768; for (y = newy; y > 0 ; y--){ if(do_rect) { rect = ibuf->rect; rect += (ofsy >> 16) * ibuf->x; } if (do_float) { rectf = (struct imbufRGBA *)ibuf->rect_float; rectf += (ofsy >> 16) * ibuf->x; } ofsy += stepy; ofsx = 32768; for (x = newx ; x>0 ; x--){ if (do_rect) *newrect++ = rect[ofsx >> 16]; if (do_float) *newrectf++ = rectf[ofsx >> 16]; ofsx += stepx; } } if (do_rect) { imb_freerectImBuf(ibuf); ibuf->mall |= IB_rect; ibuf->rect = _newrect; } if (do_float) { imb_freerectfloatImBuf(ibuf); ibuf->mall |= IB_rectfloat; ibuf->rect_float = (float *)_newrectf; } scalefast_Z_ImBuf(ibuf, newx, newy); ibuf->x = newx; ibuf->y = newy; return(ibuf); } static struct ImBuf *generic_fieldscale(struct ImBuf *ibuf, short newx, short newy, struct ImBuf *(*scalefunc)(ImBuf *, short, short) ) { struct ImBuf *sbuf1, *sbuf2; sbuf1 = IMB_allocImBuf(ibuf->x, ibuf->y / 2, ibuf->depth, ibuf->flags, 0); sbuf2 = IMB_allocImBuf(ibuf->x, ibuf->y / 2, ibuf->depth, ibuf->flags, 0); ibuf->x *= 2; /* more args needed, 0 assumed... (nzc) */ IMB_rectcpy(sbuf1, ibuf, 0, 0, 0, 0, ibuf->x, ibuf->y); IMB_rectcpy(sbuf2, ibuf, 0, 0, sbuf2->x, 0, ibuf->x, ibuf->y); imb_freerectImBuf(ibuf); imb_freerectfloatImBuf(ibuf); ibuf->x = newx; ibuf->y = newy; imb_addrectImBuf(ibuf); if(ibuf->flags & IB_rectfloat) imb_addrectfloatImBuf(ibuf); scalefunc(sbuf1, newx, newy / 2); scalefunc(sbuf2, newx, newy / 2); ibuf->x *= 2; /* more args needed, 0 assumed... (nzc) */ IMB_rectcpy(ibuf, sbuf1, 0, 0, 0, 0, sbuf1->x, sbuf1->y); IMB_rectcpy(ibuf, sbuf2, sbuf2->x, 0, 0, 0, sbuf2->x, sbuf2->y); ibuf->x /= 2; IMB_freeImBuf(sbuf1); IMB_freeImBuf(sbuf2); return(ibuf); } struct ImBuf *IMB_scalefastfieldImBuf(struct ImBuf *ibuf, short newx, short newy) { return(generic_fieldscale(ibuf, newx, newy, IMB_scalefastImBuf)); } struct ImBuf *IMB_scalefieldImBuf(struct ImBuf *ibuf, short newx, short newy) { return(generic_fieldscale(ibuf, newx, newy, IMB_scaleImBuf)); }