]> git.tokkee.org Git - inkscape.git/commitdiff

Code

pixblock-transform and scaler: convert src if needed, to match dest mode.
authorjfbarraud <jfbarraud@users.sourceforge.net>
Sun, 24 Feb 2008 01:35:24 +0000 (01:35 +0000)
committerjfbarraud <jfbarraud@users.sourceforge.net>
Sun, 24 Feb 2008 01:35:24 +0000 (01:35 +0000)
color matrix, comp. transfert and diplacement map: explicitely free some unused pixblock instead of resting on garbage collection...

src/display/nr-filter-colormatrix.cpp
src/display/nr-filter-component-transfer.cpp
src/display/nr-filter-displacement-map.cpp
src/display/pixblock-scaler.cpp
src/display/pixblock-transform.cpp

index 054b3899b41171a84156539a208a82995ca96def..79c6177ab96d45af698547182d2dd985c3869ed9 100644 (file)
@@ -43,6 +43,7 @@ int FilterColorMatrix::render(FilterSlot &slot, FilterUnits const &/*units*/) {
 
     // this primitive is defined for non-premultiplied RGBA values,
     // thus convert them to that format
+    bool free_in_on_exit = false;
     if (in->mode != NR_PIXBLOCK_MODE_R8G8B8A8N) {
         NRPixBlock *original_in = in;
         in = new NRPixBlock;
@@ -51,6 +52,7 @@ int FilterColorMatrix::render(FilterSlot &slot, FilterUnits const &/*units*/) {
                                original_in->area.x1, original_in->area.y1,
                                false);
         nr_blit_pixblock_pixblock(in, original_in);
+        free_in_on_exit = true;
     }
 
     unsigned char *in_data = NR_PIXBLOCK_PX(in);
@@ -143,6 +145,12 @@ int FilterColorMatrix::render(FilterSlot &slot, FilterUnits const &/*units*/) {
         case COLORMATRIX_ENDTYPE:
             break;
     }
+
+    if (free_in_on_exit) {
+        nr_pixblock_release(in);
+        delete in;
+    }
+
     out->empty = FALSE;
     slot.set(_output, out);
     return 0;
index 1edb3ac98fcca609f2bed031d64e1e4d441bc013..a81d8f53ab7b41a08507d65203f1a9be9c46bd6b 100644 (file)
@@ -47,6 +47,7 @@ int FilterComponentTransfer::render(FilterSlot &slot, FilterUnits const &/*units
 
     // this primitive is defined for non-premultiplied RGBA values,
     // thus convert them to that format before blending
+    bool free_in_on_exit = false;
     if (in->mode != NR_PIXBLOCK_MODE_R8G8B8A8N) {
         NRPixBlock *original_in = in;
         in = new NRPixBlock;
@@ -55,6 +56,7 @@ int FilterComponentTransfer::render(FilterSlot &slot, FilterUnits const &/*units
                                original_in->area.x1, original_in->area.y1,
                                false);
         nr_blit_pixblock_pixblock(in, original_in);
+        free_in_on_exit = true;
     }
 
     unsigned char *in_data = NR_PIXBLOCK_PX(in);
@@ -117,6 +119,11 @@ int FilterComponentTransfer::render(FilterSlot &slot, FilterUnits const &/*units
         }
     }
 
+    if (free_in_on_exit) {
+        nr_pixblock_release(in);
+        delete in;
+    }
+
     out->empty = FALSE;
     slot.set(_output, out);
     return 0;
index 1345924fcd0057a2a29c93aa5ce32060477e9997..ecf0e8defc6358d69649450ee2b50e3c598fb9f1 100644 (file)
@@ -41,18 +41,11 @@ int FilterDisplacementMap::render(FilterSlot &slot, FilterUnits const &units) {
     if (map->area.x1 <= map->area.x0 || map->area.y1 <=  map->area.y0) return 0; //nothing to do!
 
     NRPixBlock *out = new NRPixBlock;
-
-    //are these checks really necessary?
-    if (out_w > map->area.x1 - out_x0) out_w = map->area.x1 - out_x0;
-    if (out_h > map->area.y1 - out_y0) out_h = map->area.y1 - out_y0;
-    if (out_x0 < map->area.x0){
-        out_x0 = map->area.x0;
-        out_w -= (map->area.x0 - out_x0);
-    }
-    if (out_y0 < map->area.y0){
-        out_y0 = map->area.y0;
-        out_h -= (map->area.y0 - out_y0);
-    }
+    
+    out_x0 = map->area.x0;
+    out_y0 = map->area.y0;
+    out_w  = map->area.x1 - map->area.x0;
+    out_h  = map->area.y1 - map->area.y0;
 
     out->area.x0 = out_x0;
     out->area.y0 = out_y0;
@@ -63,6 +56,7 @@ int FilterDisplacementMap::render(FilterSlot &slot, FilterUnits const &units) {
 
     // this primitive is defined for non-premultiplied RGBA values,
     // thus convert them to that format
+    bool free_map_on_exit = false;
     if (map->mode != NR_PIXBLOCK_MODE_R8G8B8A8N) {
         NRPixBlock *original_map = map;
         map = new NRPixBlock;
@@ -71,6 +65,7 @@ int FilterDisplacementMap::render(FilterSlot &slot, FilterUnits const &units) {
                                original_map->area.x1, original_map->area.y1,
                                false);
         nr_blit_pixblock_pixblock(map, original_map);
+        free_map_on_exit = true;
     }
 
     unsigned char *map_data = NR_PIXBLOCK_PX(map);
@@ -87,29 +82,36 @@ int FilterDisplacementMap::render(FilterSlot &slot, FilterUnits const &units) {
     
     for (x=0; x < out_w; x++){
         for (y=0; y < out_h; y++){
-            if (x+out_x0-map->area.x0 >= 0 &&
-                x+out_x0-map->area.x0 < in_w &&
-                y+out_y0-map->area.y0 >= 0 &&
-                y+out_y0-map->area.y0 < in_h){
-
-                    coordx = out_x0 - map->area.x0 + x + scalex * ( double(map_data[4*((x+out_x0-map->area.x0) + in_w*(y+out_y0-map->area.y0)) + Xchannel])/255 - 0.5);
-                    coordy = out_y0 - map->area.y0 + y + scaley * ( double(map_data[4*((x+out_x0-map->area.x0) + in_w*(y+out_y0-map->area.y0)) + Ychannel])/255 - 0.5);
-
-                    if (coordx>=0 && coordx<in_w && coordy>=0 && coordy<in_h){
-                            out_data[4*(x + out_w*y)] = texture_data[4*(int(coordx) + int(coordy)*in_w)];
-                            out_data[4*(x + out_w*y) + 1] = texture_data[4*(int(coordx) + int(coordy)*in_w) + 1];
-                            out_data[4*(x + out_w*y) + 2] = texture_data[4*(int(coordx) + int(coordy)*in_w) + 2];
-                            out_data[4*(x + out_w*y) + 3] = texture_data[4*(int(coordx) + int(coordy)*in_w) + 3];
-                    } else {
-                            out_data[4*(x + out_w*y)] = 255;
-                            out_data[4*(x + out_w*y) + 1] = 255;
-                            out_data[4*(x + out_w*y) + 2] = 255;
-                            out_data[4*(x + out_w*y) + 3] = 0;
-                    }
+            int xmap = x+out_x0-map->area.x0;
+            int ymap = y+out_y0-map->area.y0;
+            if (xmap >= 0 &&
+                xmap < in_w &&
+                ymap >= 0 &&
+                ymap < in_h){
+
+                coordx = xmap + scalex * ( double(map_data[4*(xmap + in_w*ymap) + Xchannel]-128.)/256);
+                coordy = ymap + scaley * ( double(map_data[4*(xmap + in_w*ymap) + Ychannel]-128.)/256);
+
+                if (coordx>=0 && coordx<in_w && coordy>=0 && coordy<in_h){
+                    out_data[4*(x + out_w*y)    ] = texture_data[4*(int(coordx) + int(coordy)*in_w)    ];
+                    out_data[4*(x + out_w*y) + 1] = texture_data[4*(int(coordx) + int(coordy)*in_w) + 1];
+                    out_data[4*(x + out_w*y) + 2] = texture_data[4*(int(coordx) + int(coordy)*in_w) + 2];
+                    out_data[4*(x + out_w*y) + 3] = texture_data[4*(int(coordx) + int(coordy)*in_w) + 3];
+                } else {
+                    out_data[4*(x + out_w*y)    ] = 255;
+                    out_data[4*(x + out_w*y) + 1] = 255;
+                    out_data[4*(x + out_w*y) + 2] = 255;
+                    out_data[4*(x + out_w*y) + 3] = 0;
+                }
             }
         }
     }
 
+    if (free_map_on_exit) {
+        nr_pixblock_release(map);
+        delete map;
+    }
+
     out->empty = FALSE;
     slot.set(_output, out);
             return 0;
@@ -140,18 +142,17 @@ void FilterDisplacementMap::set_channel_selector(int s, FilterDisplacementMapCha
 
 void FilterDisplacementMap::area_enlarge(NRRectL &area, Matrix const &trans)
 {
-    out_x0 = area.x0;
-    out_y0 = area.y0;
-    out_w = area.x1 - area.x0;
-    out_h = area.y1 - area.y0;
+    //I assume scale is in user coordinates (?!?)
+    //FIXME: trans should be multiplied by some primitiveunits2user, shouldn't it?
     
-    double scalex = scale*trans.expansionX();
-    double scaley = scale*trans.expansionY();
-
-    area.x0 -= (int)(scalex/2);
-    area.x1 += (int)(scalex/2);
-    area.y0 -= (int)(scaley/2);
-    area.y1 += (int)(scaley/2);
+    double scalex = scale/2.*(std::fabs(trans[0])+std::fabs(trans[1]));
+    double scaley = scale/2.*(std::fabs(trans[2])+std::fabs(trans[3]));
+
+    //FIXME: no +2 should be there!... (noticable only for big scales at big zoom factor)
+    area.x0 -= (int)(scalex)+2;
+    area.x1 += (int)(scalex)+2;
+    area.y0 -= (int)(scaley)+2;
+    area.y1 += (int)(scaley)+2;
 }
 
 FilterTraits FilterDisplacementMap::get_input_traits() {
index 5bf3c2762eb7c93884f9cbefb326ab3085d0fe7d..39c82daf6ed9ba5808f70514afd1b85b4c6efacc 100644 (file)
@@ -96,11 +96,13 @@ static void scale_bicubic_rgba(NRPixBlock *to, NRPixBlock *from)
         return;
     }
 
+    bool free_from_on_exit = false;
     if (from->mode != to->mode){
         NRPixBlock *o_from = from;
         from = new NRPixBlock;
         nr_pixblock_setup_fast(from, to->mode, o_from->area.x0, o_from->area.y0, o_from->area.x1, o_from->area.y1, false);
         nr_blit_pixblock_pixblock(from, o_from);
+        free_from_on_exit = true;
     }
 
     // Precalculate sizes of source and destination pixblocks
@@ -198,6 +200,11 @@ static void scale_bicubic_rgba(NRPixBlock *to, NRPixBlock *from)
             }
         }
     }
+    if (free_from_on_exit) {
+        nr_pixblock_release(from);
+        delete from;
+    }
+
 }
 
 void scale_bicubic_alpha(NRPixBlock *to, NRPixBlock *from)
index abe0f90c0f9aa86cbb48fb10926f04f8697b9982..730f0ad4318e1987678b09e8a13bef13bb72a1d4 100644 (file)
@@ -52,6 +52,15 @@ void transform_nearest(NRPixBlock *to, NRPixBlock *from, Matrix &trans)
         return;
     }
 
+    bool free_from_on_exit = false;
+    if (from->mode != to->mode){
+        NRPixBlock *o_from = from;
+        from = new NRPixBlock;
+        nr_pixblock_setup_fast(from, to->mode, o_from->area.x0, o_from->area.y0, o_from->area.x1, o_from->area.y1, false);
+        nr_blit_pixblock_pixblock(from, o_from);
+        free_from_on_exit = true;
+    }
+
     // Precalculate sizes of source and destination pixblocks
     int from_width = from->area.x1 - from->area.x0;
     int from_height = from->area.y1 - from->area.y0;
@@ -90,6 +99,10 @@ void transform_nearest(NRPixBlock *to, NRPixBlock *from, Matrix &trans)
             NR_PIXBLOCK_PX(to)[to_y * to->rs + to_x * 4 + 3] = result.a;
         }
     }
+    if (free_from_on_exit) {
+        nr_pixblock_release(from);
+        delete from;
+    }
 }
 
 /** Calculates cubically interpolated value of the four given pixel values.
@@ -149,11 +162,13 @@ void transform_bicubic(NRPixBlock *to, NRPixBlock *from, Matrix &trans)
         return;
     }
 
+    bool free_from_on_exit = false;
     if (from->mode != to->mode){
         NRPixBlock *o_from = from;
         from = new NRPixBlock;
         nr_pixblock_setup_fast(from, to->mode, o_from->area.x0, o_from->area.y0, o_from->area.x1, o_from->area.y1, false);
         nr_blit_pixblock_pixblock(from, o_from);
+        free_from_on_exit = true;
     }
     
     // Precalculate sizes of source and destination pixblocks
@@ -255,6 +270,10 @@ void transform_bicubic(NRPixBlock *to, NRPixBlock *from, Matrix &trans)
             }
         }
     }
+    if (free_from_on_exit) {
+        nr_pixblock_release(from);
+        delete from;
+    }
 }
 
 } /* namespace NR */