1 /*
2 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
3 % %
4 % %
5 % %
6 % PPPP IIIII X X EEEEE L %
7 % P P I X X E L %
8 % PPPP I X EEE L %
9 % P I X X E L %
10 % P IIIII X X EEEEE LLLLL %
11 % %
12 % IIIII TTTTT EEEEE RRRR AAA TTTTT OOO RRRR %
13 % I T E R R A A T O O R R %
14 % I T EEE RRRR AAAAA T O O RRRR %
15 % I T E R R A A T O O R R %
16 % IIIII T EEEEE R R A A T OOO R R %
17 % %
18 % %
19 % ImageMagick Image Pixel Iterator Methods %
20 % %
21 % Software Design %
22 % Cristy %
23 % March 2003 %
24 % %
25 % %
26 % Copyright 1999-2020 ImageMagick Studio LLC, a non-profit organization %
27 % dedicated to making software imaging solutions freely available. %
28 % %
29 % You may not use this file except in compliance with the License. You may %
30 % obtain a copy of the License at %
31 % %
32 % https://imagemagick.org/script/license.php %
33 % %
34 % Unless required by applicable law or agreed to in writing, software %
35 % distributed under the License is distributed on an "AS IS" BASIS, %
36 % WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. %
37 % See the License for the specific language governing permissions and %
38 % limitations under the License. %
39 % %
40 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
41 %
42 %
43 %
44 */
45
46 /*
47 Include declarations.
48 */
49 #include "MagickWand/studio.h"
50 #include "MagickWand/MagickWand.h"
51 #include "MagickWand/magick-wand-private.h"
52 #include "MagickWand/pixel-iterator.h"
53 #include "MagickWand/pixel-wand.h"
54 #include "MagickWand/wand.h"
55
56 /*
57 Define declarations.
58 */
59 #define PixelIteratorId "PixelIterator"
60
61 /*
62 Typedef declarations.
63 */
64 struct _PixelIterator
65 {
66 size_t
67 id;
68
69 char
70 name[MagickPathExtent];
71
72 ExceptionInfo
73 *exception;
74
75 CacheView
76 *view;
77
78 RectangleInfo
79 region;
80
81 MagickBooleanType
82 active; /* user has been given pixel data */
83
84 ssize_t
85 y;
86
87 PixelWand
88 **pixel_wands;
89
90 MagickBooleanType
91 debug;
92
93 size_t
94 signature;
95 };
96
97 /*
98 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
99 % %
100 % %
101 % %
102 % C l e a r P i x e l I t e r a t o r %
103 % %
104 % %
105 % %
106 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
107 %
108 % ClearPixelIterator() clear resources associated with a PixelIterator.
109 %
110 % The format of the ClearPixelIterator method is:
111 %
112 % void ClearPixelIterator(PixelIterator *iterator)
113 %
114 % A description of each parameter follows:
115 %
116 % o iterator: the pixel iterator.
117 %
118 */
ClearPixelIterator(PixelIterator * iterator)119 WandExport void ClearPixelIterator(PixelIterator *iterator)
120 {
121 assert(iterator != (const PixelIterator *) NULL);
122 assert(iterator->signature == MagickWandSignature);
123 if (iterator->debug != MagickFalse)
124 (void) LogMagickEvent(WandEvent,GetMagickModule(),"%s",iterator->name);
125 iterator->pixel_wands=DestroyPixelWands(iterator->pixel_wands,
126 iterator->region.width);
127 ClearMagickException(iterator->exception);
128 iterator->pixel_wands=NewPixelWands(iterator->region.width);
129 iterator->active=MagickFalse;
130 iterator->y=0;
131 iterator->debug=IsEventLogging();
132 }
133
134 /*
135 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
136 % %
137 % %
138 % %
139 % C l o n e P i x e l I t e r a t o r %
140 % %
141 % %
142 % %
143 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
144 %
145 % ClonePixelIterator() makes an exact copy of the specified iterator.
146 %
147 % The format of the ClonePixelIterator method is:
148 %
149 % PixelIterator *ClonePixelIterator(const PixelIterator *iterator)
150 %
151 % A description of each parameter follows:
152 %
153 % o iterator: the magick iterator.
154 %
155 */
ClonePixelIterator(const PixelIterator * iterator)156 WandExport PixelIterator *ClonePixelIterator(const PixelIterator *iterator)
157 {
158 PixelIterator
159 *clone_iterator;
160
161 assert(iterator != (PixelIterator *) NULL);
162 assert(iterator->signature == MagickWandSignature);
163 if (iterator->debug != MagickFalse)
164 (void) LogMagickEvent(WandEvent,GetMagickModule(),"%s",iterator->name);
165 clone_iterator=(PixelIterator *) AcquireMagickMemory(sizeof(*clone_iterator));
166 if (clone_iterator == (PixelIterator *) NULL)
167 ThrowWandFatalException(ResourceLimitFatalError,"MemoryAllocationFailed",
168 iterator->name);
169 (void) memset(clone_iterator,0,sizeof(*clone_iterator));
170 clone_iterator->id=AcquireWandId();
171 (void) FormatLocaleString(clone_iterator->name,MagickPathExtent,"%s-%.20g",
172 PixelIteratorId,(double) clone_iterator->id);
173 clone_iterator->exception=AcquireExceptionInfo();
174 InheritException(clone_iterator->exception,iterator->exception);
175 clone_iterator->view=CloneCacheView(iterator->view);
176 clone_iterator->region=iterator->region;
177 clone_iterator->active=iterator->active;
178 clone_iterator->y=iterator->y;
179 clone_iterator->pixel_wands=ClonePixelWands((const PixelWand **)
180 iterator->pixel_wands,iterator->region.width);
181 clone_iterator->debug=iterator->debug;
182 if (clone_iterator->debug != MagickFalse)
183 (void) LogMagickEvent(WandEvent,GetMagickModule(),"%s",
184 clone_iterator->name);
185 clone_iterator->signature=MagickWandSignature;
186 return(clone_iterator);
187 }
188
189 /*
190 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
191 % %
192 % %
193 % %
194 % D e s t r o y P i x e l I t e r a t o r %
195 % %
196 % %
197 % %
198 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
199 %
200 % DestroyPixelIterator() deallocates resources associated with a PixelIterator.
201 %
202 % The format of the DestroyPixelIterator method is:
203 %
204 % PixelIterator *DestroyPixelIterator(PixelIterator *iterator)
205 %
206 % A description of each parameter follows:
207 %
208 % o iterator: the pixel iterator.
209 %
210 */
DestroyPixelIterator(PixelIterator * iterator)211 WandExport PixelIterator *DestroyPixelIterator(PixelIterator *iterator)
212 {
213 assert(iterator != (const PixelIterator *) NULL);
214 assert(iterator->signature == MagickWandSignature);
215 if (iterator->debug != MagickFalse)
216 (void) LogMagickEvent(WandEvent,GetMagickModule(),"%s",iterator->name);
217 iterator->view=DestroyCacheView(iterator->view);
218 iterator->pixel_wands=DestroyPixelWands(iterator->pixel_wands,
219 iterator->region.width);
220 iterator->exception=DestroyExceptionInfo(iterator->exception);
221 iterator->signature=(~MagickWandSignature);
222 RelinquishWandId(iterator->id);
223 iterator=(PixelIterator *) RelinquishMagickMemory(iterator);
224 return(iterator);
225 }
226
227 /*
228 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
229 % %
230 % %
231 % %
232 % I s P i x e l I t e r a t o r %
233 % %
234 % %
235 % %
236 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
237 %
238 % IsPixelIterator() returns MagickTrue if the iterator is verified as a pixel
239 % iterator.
240 %
241 % The format of the IsPixelIterator method is:
242 %
243 % MagickBooleanType IsPixelIterator(const PixelIterator *iterator)
244 %
245 % A description of each parameter follows:
246 %
247 % o iterator: the magick iterator.
248 %
249 */
IsPixelIterator(const PixelIterator * iterator)250 WandExport MagickBooleanType IsPixelIterator(const PixelIterator *iterator)
251 {
252 size_t
253 length;
254
255 if (iterator == (const PixelIterator *) NULL)
256 return(MagickFalse);
257 if (iterator->signature != MagickWandSignature)
258 return(MagickFalse);
259 length=strlen(PixelIteratorId);
260 if (LocaleNCompare(iterator->name,PixelIteratorId,length) != 0)
261 return(MagickFalse);
262 return(MagickTrue);
263 }
264
265 /*
266 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
267 % %
268 % %
269 % %
270 % N e w P i x e l I t e r a t o r %
271 % %
272 % %
273 % %
274 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
275 %
276 % NewPixelIterator() returns a new pixel iterator.
277 %
278 % The format of the NewPixelIterator method is:
279 %
280 % PixelIterator *NewPixelIterator(MagickWand *wand)
281 %
282 % A description of each parameter follows:
283 %
284 % o wand: the magick wand.
285 %
286 */
NewPixelIterator(MagickWand * wand)287 WandExport PixelIterator *NewPixelIterator(MagickWand *wand)
288 {
289 const char
290 *quantum;
291
292 ExceptionInfo
293 *exception;
294
295 Image
296 *image;
297
298 PixelIterator
299 *iterator;
300
301 size_t
302 depth;
303
304 CacheView
305 *view;
306
307 depth=MAGICKCORE_QUANTUM_DEPTH;
308 quantum=GetMagickQuantumDepth(&depth);
309 if (depth != MAGICKCORE_QUANTUM_DEPTH)
310 ThrowWandFatalException(WandError,"QuantumDepthMismatch",quantum);
311 assert(wand != (MagickWand *) NULL);
312 image=GetImageFromMagickWand(wand);
313 if (image == (Image *) NULL)
314 return((PixelIterator *) NULL);
315 exception=AcquireExceptionInfo();
316 view=AcquireVirtualCacheView(image,exception);
317 if (view == (CacheView *) NULL)
318 return((PixelIterator *) NULL);
319 iterator=(PixelIterator *) AcquireMagickMemory(sizeof(*iterator));
320 if (iterator == (PixelIterator *) NULL)
321 ThrowWandFatalException(ResourceLimitFatalError,"MemoryAllocationFailed",
322 GetExceptionMessage(errno));
323 (void) memset(iterator,0,sizeof(*iterator));
324 iterator->id=AcquireWandId();
325 (void) FormatLocaleString(iterator->name,MagickPathExtent,"%s-%.20g",
326 PixelIteratorId,(double) iterator->id);
327 iterator->exception=exception;
328 iterator->view=view;
329 SetGeometry(image,&iterator->region);
330 iterator->region.width=image->columns;
331 iterator->region.height=image->rows;
332 iterator->region.x=0;
333 iterator->region.y=0;
334 iterator->pixel_wands=NewPixelWands(iterator->region.width);
335 iterator->y=0;
336 iterator->debug=IsEventLogging();
337 if (iterator->debug != MagickFalse)
338 (void) LogMagickEvent(WandEvent,GetMagickModule(),"%s",iterator->name);
339 iterator->signature=MagickWandSignature;
340 return(iterator);
341 }
342
343 /*
344 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
345 % %
346 % %
347 % %
348 % P i x e l C l e a r I t e r a t o r E x c e p t i o n %
349 % %
350 % %
351 % %
352 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
353 %
354 % PixelClearIteratorException() clear any exceptions associated with the
355 % iterator.
356 %
357 % The format of the PixelClearIteratorException method is:
358 %
359 % MagickBooleanType PixelClearIteratorException(PixelIterator *iterator)
360 %
361 % A description of each parameter follows:
362 %
363 % o iterator: the pixel iterator.
364 %
365 */
PixelClearIteratorException(PixelIterator * iterator)366 WandExport MagickBooleanType PixelClearIteratorException(
367 PixelIterator *iterator)
368 {
369 assert(iterator != (PixelIterator *) NULL);
370 assert(iterator->signature == MagickWandSignature);
371 if (iterator->debug != MagickFalse)
372 (void) LogMagickEvent(WandEvent,GetMagickModule(),"%s",iterator->name);
373 ClearMagickException(iterator->exception);
374 return(MagickTrue);
375 }
376
377 /*
378 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
379 % %
380 % %
381 % %
382 % N e w P i x e l R e g i o n I t e r a t o r %
383 % %
384 % %
385 % %
386 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
387 %
388 % NewPixelRegionIterator() returns a new pixel iterator.
389 %
390 % The format of the NewPixelRegionIterator method is:
391 %
392 % PixelIterator *NewPixelRegionIterator(MagickWand *wand,const ssize_t x,
393 % const ssize_t y,const size_t width,const size_t height)
394 %
395 % A description of each parameter follows:
396 %
397 % o wand: the magick wand.
398 %
399 % o x,y,columns,rows: These values define the perimeter of a region of
400 % pixels.
401 %
402 */
NewPixelRegionIterator(MagickWand * wand,const ssize_t x,const ssize_t y,const size_t width,const size_t height)403 WandExport PixelIterator *NewPixelRegionIterator(MagickWand *wand,
404 const ssize_t x,const ssize_t y,const size_t width,const size_t height)
405 {
406 CacheView
407 *view;
408
409 const char
410 *quantum;
411
412 ExceptionInfo
413 *exception;
414
415 Image
416 *image;
417
418 PixelIterator
419 *iterator;
420
421 size_t
422 depth;
423
424 assert(wand != (MagickWand *) NULL);
425 depth=MAGICKCORE_QUANTUM_DEPTH;
426 quantum=GetMagickQuantumDepth(&depth);
427 if (depth != MAGICKCORE_QUANTUM_DEPTH)
428 ThrowWandFatalException(WandError,"QuantumDepthMismatch",quantum);
429 if ((width == 0) || (height == 0))
430 ThrowWandFatalException(WandError,"ZeroRegionSize",quantum);
431 image=GetImageFromMagickWand(wand);
432 if (image == (Image *) NULL)
433 return((PixelIterator *) NULL);
434 exception=AcquireExceptionInfo();
435 view=AcquireVirtualCacheView(image,exception);
436 if (view == (CacheView *) NULL)
437 return((PixelIterator *) NULL);
438 iterator=(PixelIterator *) AcquireMagickMemory(sizeof(*iterator));
439 if (iterator == (PixelIterator *) NULL)
440 ThrowWandFatalException(ResourceLimitFatalError,"MemoryAllocationFailed",
441 wand->name);
442 (void) memset(iterator,0,sizeof(*iterator));
443 iterator->id=AcquireWandId();
444 (void) FormatLocaleString(iterator->name,MagickPathExtent,"%s-%.20g",
445 PixelIteratorId,(double) iterator->id);
446 iterator->exception=exception;
447 iterator->view=view;
448 SetGeometry(image,&iterator->region);
449 iterator->region.width=width;
450 iterator->region.height=height;
451 iterator->region.x=x;
452 iterator->region.y=y;
453 iterator->pixel_wands=NewPixelWands(iterator->region.width);
454 iterator->y=0;
455 iterator->debug=IsEventLogging();
456 if (iterator->debug != MagickFalse)
457 (void) LogMagickEvent(WandEvent,GetMagickModule(),"%s",iterator->name);
458 iterator->signature=MagickWandSignature;
459 return(iterator);
460 }
461
462 /*
463 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
464 % %
465 % %
466 % %
467 % P i x e l G e t C u r r e n t I t e r a t o r R o w %
468 % %
469 % %
470 % %
471 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
472 %
473 % PixelGetCurrentIteratorRow() returns the current row as an array of pixel
474 % wands from the pixel iterator.
475 %
476 % The format of the PixelGetCurrentIteratorRow method is:
477 %
478 % PixelWand **PixelGetCurrentIteratorRow(PixelIterator *iterator,
479 % size_t *number_wands)
480 %
481 % A description of each parameter follows:
482 %
483 % o iterator: the pixel iterator.
484 %
485 % o number_wands: the number of pixel wands.
486 %
487 */
PixelGetCurrentIteratorRow(PixelIterator * iterator,size_t * number_wands)488 WandExport PixelWand **PixelGetCurrentIteratorRow(PixelIterator *iterator,
489 size_t *number_wands)
490 {
491 register const Quantum
492 *pixels;
493
494 register ssize_t
495 x;
496
497 assert(iterator != (PixelIterator *) NULL);
498 assert(iterator->signature == MagickWandSignature);
499 if (iterator->debug != MagickFalse)
500 (void) LogMagickEvent(WandEvent,GetMagickModule(),"%s",iterator->name);
501 *number_wands=0;
502 iterator->active=MagickTrue;
503 pixels=GetCacheViewVirtualPixels(iterator->view,iterator->region.x,
504 iterator->region.y+iterator->y,iterator->region.width,1,
505 iterator->exception);
506 if (pixels == (const Quantum *) NULL)
507 return((PixelWand **) NULL);
508 for (x=0; x < (ssize_t) iterator->region.width; x++)
509 {
510 PixelSetQuantumPixel(GetCacheViewImage(iterator->view),pixels,
511 iterator->pixel_wands[x]);
512 pixels+=GetPixelChannels(GetCacheViewImage(iterator->view));
513 }
514 *number_wands=iterator->region.width;
515 return(iterator->pixel_wands);
516 }
517
518 /*
519 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
520 % %
521 % %
522 % %
523 % P i x e l G e t I t e r a t o r E x c e p t i o n %
524 % %
525 % %
526 % %
527 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
528 %
529 % PixelGetIteratorException() returns the severity, reason, and description of
530 % any error that occurs when using other methods in this API.
531 %
532 % The format of the PixelGetIteratorException method is:
533 %
534 % char *PixelGetIteratorException(const PixelIterator *iterator,
535 % ExceptionType *severity)
536 %
537 % A description of each parameter follows:
538 %
539 % o iterator: the pixel iterator.
540 %
541 % o severity: the severity of the error is returned here.
542 %
543 */
PixelGetIteratorException(const PixelIterator * iterator,ExceptionType * severity)544 WandExport char *PixelGetIteratorException(const PixelIterator *iterator,
545 ExceptionType *severity)
546 {
547 char
548 *description;
549
550 assert(iterator != (const PixelIterator *) NULL);
551 assert(iterator->signature == MagickWandSignature);
552 if (iterator->debug != MagickFalse)
553 (void) LogMagickEvent(WandEvent,GetMagickModule(),"%s",iterator->name);
554 assert(severity != (ExceptionType *) NULL);
555 *severity=iterator->exception->severity;
556 description=(char *) AcquireQuantumMemory(2UL*MagickPathExtent,
557 sizeof(*description));
558 if (description == (char *) NULL)
559 ThrowWandFatalException(ResourceLimitFatalError,"MemoryAllocationFailed",
560 iterator->name);
561 *description='\0';
562 if (iterator->exception->reason != (char *) NULL)
563 (void) CopyMagickString(description,GetLocaleExceptionMessage(
564 iterator->exception->severity,iterator->exception->reason),MagickPathExtent);
565 if (iterator->exception->description != (char *) NULL)
566 {
567 (void) ConcatenateMagickString(description," (",MagickPathExtent);
568 (void) ConcatenateMagickString(description,GetLocaleExceptionMessage(
569 iterator->exception->severity,iterator->exception->description),
570 MagickPathExtent);
571 (void) ConcatenateMagickString(description,")",MagickPathExtent);
572 }
573 return(description);
574 }
575
576 /*
577 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
578 % %
579 % %
580 % %
581 % P i x e l G e t E x c e p t i o n T y p e %
582 % %
583 % %
584 % %
585 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
586 %
587 % PixelGetIteratorExceptionType() the exception type associated with the
588 % iterator. If no exception has occurred, UndefinedExceptionType is returned.
589 %
590 % The format of the PixelGetIteratorExceptionType method is:
591 %
592 % ExceptionType PixelGetIteratorExceptionType(
593 % const PixelIterator *iterator)
594 %
595 % A description of each parameter follows:
596 %
597 % o iterator: the pixel iterator.
598 %
599 */
PixelGetIteratorExceptionType(const PixelIterator * iterator)600 WandExport ExceptionType PixelGetIteratorExceptionType(
601 const PixelIterator *iterator)
602 {
603 assert(iterator != (const PixelIterator *) NULL);
604 assert(iterator->signature == MagickWandSignature);
605 if (iterator->debug != MagickFalse)
606 (void) LogMagickEvent(WandEvent,GetMagickModule(),"%s",iterator->name);
607 return(iterator->exception->severity);
608 }
609
610 /*
611 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
612 % %
613 % %
614 % %
615 % P i x e l G e t I t e r a t o r R o w %
616 % %
617 % %
618 % %
619 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
620 %
621 % PixelGetIteratorRow() returns the current pixel iterator row.
622 %
623 % The format of the PixelGetIteratorRow method is:
624 %
625 % MagickBooleanType PixelGetIteratorRow(PixelIterator *iterator)
626 %
627 % A description of each parameter follows:
628 %
629 % o iterator: the pixel iterator.
630 %
631 */
PixelGetIteratorRow(PixelIterator * iterator)632 WandExport ssize_t PixelGetIteratorRow(PixelIterator *iterator)
633 {
634 assert(iterator != (const PixelIterator *) NULL);
635 assert(iterator->signature == MagickWandSignature);
636 if (iterator->debug != MagickFalse)
637 (void) LogMagickEvent(WandEvent,GetMagickModule(),"%s",iterator->name);
638 return(iterator->y);
639 }
640
641 /*
642 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
643 % %
644 % %
645 % %
646 % P i x e l G e t N e x t I t e r a t o r R o w %
647 % %
648 % %
649 % %
650 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
651 %
652 % PixelGetNextIteratorRow() returns the next row as an array of pixel wands
653 % from the pixel iterator.
654 %
655 % The format of the PixelGetNextIteratorRow method is:
656 %
657 % PixelWand **PixelGetNextIteratorRow(PixelIterator *iterator,
658 % size_t *number_wands)
659 %
660 % A description of each parameter follows:
661 %
662 % o iterator: the pixel iterator.
663 %
664 % o number_wands: the number of pixel wands.
665 %
666 */
PixelGetNextIteratorRow(PixelIterator * iterator,size_t * number_wands)667 WandExport PixelWand **PixelGetNextIteratorRow(PixelIterator *iterator,
668 size_t *number_wands)
669 {
670 register const Quantum
671 *pixels;
672
673 register ssize_t
674 x;
675
676 assert(iterator != (PixelIterator *) NULL);
677 assert(iterator->signature == MagickWandSignature);
678 if (iterator->debug != MagickFalse)
679 (void) LogMagickEvent(WandEvent,GetMagickModule(),"%s",iterator->name);
680 *number_wands=0;
681 if (iterator->active != MagickFalse)
682 iterator->y++;
683 if (PixelSetIteratorRow(iterator,iterator->y) == MagickFalse)
684 return((PixelWand **) NULL);
685 pixels=GetCacheViewVirtualPixels(iterator->view,iterator->region.x,
686 iterator->region.y+iterator->y,iterator->region.width,1,
687 iterator->exception);
688 if (pixels == (const Quantum *) NULL)
689 return((PixelWand **) NULL);
690 for (x=0; x < (ssize_t) iterator->region.width; x++)
691 {
692 PixelSetQuantumPixel(GetCacheViewImage(iterator->view),pixels,
693 iterator->pixel_wands[x]);
694 pixels+=GetPixelChannels(GetCacheViewImage(iterator->view));
695 }
696 *number_wands=iterator->region.width;
697 return(iterator->pixel_wands);
698 }
699
700 /*
701 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
702 % %
703 % %
704 % %
705 % P i x e l G e t P r e v i o u s I t e r a t o r R o w %
706 % %
707 % %
708 % %
709 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
710 %
711 % PixelGetPreviousIteratorRow() returns the previous row as an array of pixel
712 % wands from the pixel iterator.
713 %
714 % The format of the PixelGetPreviousIteratorRow method is:
715 %
716 % PixelWand **PixelGetPreviousIteratorRow(PixelIterator *iterator,
717 % size_t *number_wands)
718 %
719 % A description of each parameter follows:
720 %
721 % o iterator: the pixel iterator.
722 %
723 % o number_wands: the number of pixel wands.
724 %
725 */
PixelGetPreviousIteratorRow(PixelIterator * iterator,size_t * number_wands)726 WandExport PixelWand **PixelGetPreviousIteratorRow(PixelIterator *iterator,
727 size_t *number_wands)
728 {
729 register const Quantum
730 *pixels;
731
732 register ssize_t
733 x;
734
735 assert(iterator != (PixelIterator *) NULL);
736 assert(iterator->signature == MagickWandSignature);
737 if (iterator->debug != MagickFalse)
738 (void) LogMagickEvent(WandEvent,GetMagickModule(),"%s",iterator->name);
739 *number_wands=0;
740 if (iterator->active != MagickFalse)
741 iterator->y--;
742 if (PixelSetIteratorRow(iterator,iterator->y) == MagickFalse)
743 return((PixelWand **) NULL);
744 pixels=GetCacheViewVirtualPixels(iterator->view,iterator->region.x,
745 iterator->region.y+iterator->y,iterator->region.width,1,
746 iterator->exception);
747 if (pixels == (const Quantum *) NULL)
748 return((PixelWand **) NULL);
749 for (x=0; x < (ssize_t) iterator->region.width; x++)
750 {
751 PixelSetQuantumPixel(GetCacheViewImage(iterator->view),pixels,
752 iterator->pixel_wands[x]);
753 pixels+=GetPixelChannels(GetCacheViewImage(iterator->view));
754 }
755 *number_wands=iterator->region.width;
756 return(iterator->pixel_wands);
757 }
758
759 /*
760 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
761 % %
762 % %
763 % %
764 % P i x e l R e s e t I t e r a t o r %
765 % %
766 % %
767 % %
768 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
769 %
770 % PixelResetIterator() resets the pixel iterator. Use it in conjunction
771 % with PixelGetNextIteratorRow() to iterate over all the pixels in a pixel
772 % container.
773 %
774 % The format of the PixelResetIterator method is:
775 %
776 % void PixelResetIterator(PixelIterator *iterator)
777 %
778 % A description of each parameter follows:
779 %
780 % o iterator: the pixel iterator.
781 %
782 */
PixelResetIterator(PixelIterator * iterator)783 WandExport void PixelResetIterator(PixelIterator *iterator)
784 {
785 assert(iterator != (PixelIterator *) NULL);
786 assert(iterator->signature == MagickWandSignature);
787 if (iterator->debug != MagickFalse)
788 (void) LogMagickEvent(WandEvent,GetMagickModule(),"%s",iterator->name);
789 iterator->active=MagickFalse;
790 iterator->y=0;
791 }
792
793 /*
794 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
795 % %
796 % %
797 % %
798 % P i x e l S e t F i r s t I t e r a t o r R o w %
799 % %
800 % %
801 % %
802 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
803 %
804 % PixelSetFirstIteratorRow() sets the pixel iterator to the first pixel row.
805 %
806 % The format of the PixelSetFirstIteratorRow method is:
807 %
808 % void PixelSetFirstIteratorRow(PixelIterator *iterator)
809 %
810 % A description of each parameter follows:
811 %
812 % o iterator: the magick iterator.
813 %
814 */
PixelSetFirstIteratorRow(PixelIterator * iterator)815 WandExport void PixelSetFirstIteratorRow(PixelIterator *iterator)
816 {
817 assert(iterator != (PixelIterator *) NULL);
818 assert(iterator->signature == MagickWandSignature);
819 if (iterator->debug != MagickFalse)
820 (void) LogMagickEvent(WandEvent,GetMagickModule(),"%s",iterator->name);
821 iterator->active=MagickFalse;
822 iterator->y=iterator->region.y;
823 }
824
825 /*
826 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
827 % %
828 % %
829 % %
830 % P i x e l S e t I t e r a t o r R o w %
831 % %
832 % %
833 % %
834 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
835 %
836 % PixelSetIteratorRow() set the pixel iterator row.
837 %
838 % The format of the PixelSetIteratorRow method is:
839 %
840 % MagickBooleanType PixelSetIteratorRow(PixelIterator *iterator,
841 % const ssize_t row)
842 %
843 % A description of each parameter follows:
844 %
845 % o iterator: the pixel iterator.
846 %
847 */
PixelSetIteratorRow(PixelIterator * iterator,const ssize_t row)848 WandExport MagickBooleanType PixelSetIteratorRow(PixelIterator *iterator,
849 const ssize_t row)
850 {
851 assert(iterator != (const PixelIterator *) NULL);
852 assert(iterator->signature == MagickWandSignature);
853 if (iterator->debug != MagickFalse)
854 (void) LogMagickEvent(WandEvent,GetMagickModule(),"%s",iterator->name);
855 if ((row < 0) || (row >= (ssize_t) iterator->region.height))
856 return(MagickFalse);
857 iterator->active=MagickTrue;
858 iterator->y=row;
859 return(MagickTrue);
860 }
861
862 /*
863 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
864 % %
865 % %
866 % %
867 % P i x e l S e t L a s t I t e r a t o r R o w %
868 % %
869 % %
870 % %
871 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
872 %
873 % PixelSetLastIteratorRow() sets the pixel iterator to the last pixel row.
874 %
875 % The format of the PixelSetLastIteratorRow method is:
876 %
877 % void PixelSetLastIteratorRow(PixelIterator *iterator)
878 %
879 % A description of each parameter follows:
880 %
881 % o iterator: the magick iterator.
882 %
883 */
PixelSetLastIteratorRow(PixelIterator * iterator)884 WandExport void PixelSetLastIteratorRow(PixelIterator *iterator)
885 {
886 assert(iterator != (PixelIterator *) NULL);
887 assert(iterator->signature == MagickWandSignature);
888 if (iterator->debug != MagickFalse)
889 (void) LogMagickEvent(WandEvent,GetMagickModule(),"%s",iterator->name);
890 iterator->active=MagickFalse;
891 iterator->y=(ssize_t) iterator->region.height-1;
892 }
893
894 /*
895 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
896 % %
897 % %
898 % %
899 % P i x e l S y n c I t e r a t o r %
900 % %
901 % %
902 % %
903 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
904 %
905 % PixelSyncIterator() syncs the pixel iterator.
906 %
907 % The format of the PixelSyncIterator method is:
908 %
909 % MagickBooleanType PixelSyncIterator(PixelIterator *iterator)
910 %
911 % A description of each parameter follows:
912 %
913 % o iterator: the pixel iterator.
914 %
915 */
PixelSyncIterator(PixelIterator * iterator)916 WandExport MagickBooleanType PixelSyncIterator(PixelIterator *iterator)
917 {
918 MagickBooleanType
919 status;
920
921 register Quantum
922 *magick_restrict pixels;
923
924 register ssize_t
925 x;
926
927 assert(iterator != (const PixelIterator *) NULL);
928 assert(iterator->signature == MagickWandSignature);
929 if (iterator->debug != MagickFalse)
930 (void) LogMagickEvent(WandEvent,GetMagickModule(),"%s",iterator->name);
931 status=SetCacheViewStorageClass(iterator->view,DirectClass,
932 iterator->exception);
933 if (status == MagickFalse)
934 return(MagickFalse);
935 pixels=GetCacheViewAuthenticPixels(iterator->view,iterator->region.x,
936 iterator->region.y+iterator->y,iterator->region.width,1,
937 iterator->exception);
938 if (pixels == (Quantum *) NULL)
939 return(MagickFalse);
940 for (x=0; x < (ssize_t) iterator->region.width; x++)
941 {
942 PixelGetQuantumPixel(GetCacheViewImage(iterator->view),
943 iterator->pixel_wands[x],pixels);
944 pixels+=GetPixelChannels(GetCacheViewImage(iterator->view));
945 }
946 if (SyncCacheViewAuthenticPixels(iterator->view,iterator->exception) == MagickFalse)
947 return(MagickFalse);
948 return(MagickTrue);
949 }
950