/* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % % BBBB L OOO BBBB % % B B L O O B B % % BBBB L O O BBBB % % B B L O O B B % % BBBB LLLLL OOO BBBB % % % % % % MagickCore Binary Large OBjectS Methods % % % % Software Design % % Cristy % % July 1999 % % % % % % Copyright 1999-2021 ImageMagick Studio LLC, a non-profit organization % % dedicated to making software imaging solutions freely available. % % % % You may not use this file except in compliance with the License. You may % % obtain a copy of the License at % % % % https://imagemagick.org/script/license.php % % % % Unless required by applicable law or agreed to in writing, software % % distributed under the License is distributed on an "AS IS" BASIS, % % WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. % % See the License for the specific language governing permissions and % % limitations under the License. % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % */ /* Include declarations. */ #ifdef __VMS #include #include #endif #include "MagickCore/studio.h" #include "MagickCore/blob.h" #include "MagickCore/blob-private.h" #include "MagickCore/cache.h" #include "MagickCore/client.h" #include "MagickCore/constitute.h" #include "MagickCore/delegate.h" #include "MagickCore/exception.h" #include "MagickCore/exception-private.h" #include "MagickCore/geometry.h" #include "MagickCore/image-private.h" #include "MagickCore/list.h" #include "MagickCore/locale_.h" #include "MagickCore/log.h" #include "MagickCore/magick.h" #include "MagickCore/memory_.h" #include "MagickCore/memory-private.h" #include "MagickCore/nt-base-private.h" #include "MagickCore/option.h" #include "MagickCore/policy.h" #include "MagickCore/resource_.h" #include "MagickCore/semaphore.h" #include "MagickCore/string_.h" #include "MagickCore/string-private.h" #include "MagickCore/timer-private.h" #include "MagickCore/token.h" #include "MagickCore/utility.h" #include "MagickCore/utility-private.h" #if defined(MAGICKCORE_ZLIB_DELEGATE) #include "zlib.h" #endif #if defined(MAGICKCORE_BZLIB_DELEGATE) #include "bzlib.h" #endif /* Define declarations. */ #define MagickMaxBlobExtent (8*8192) #if !defined(MAP_ANONYMOUS) && defined(MAP_ANON) # define MAP_ANONYMOUS MAP_ANON #endif #if !defined(MAP_FAILED) #define MAP_FAILED ((void *) -1) #endif #if defined(__OS2__) #include #define _O_BINARY O_BINARY #endif /* Typedef declarations. */ typedef union FileInfo { FILE *file; #if defined(MAGICKCORE_ZLIB_DELEGATE) gzFile gzfile; #endif #if defined(MAGICKCORE_BZLIB_DELEGATE) BZFILE *bzfile; #endif } FileInfo; struct _BlobInfo { size_t length, extent, quantum; BlobMode mode; MagickBooleanType mapped, eof; int error, error_number; MagickOffsetType offset; MagickSizeType size; MagickBooleanType exempt, synchronize, status, temporary; StreamType type; FileInfo file_info; struct stat properties; StreamHandler stream; CustomStreamInfo *custom_stream; unsigned char *data; MagickBooleanType debug; SemaphoreInfo *semaphore; ssize_t reference_count; size_t signature; }; struct _CustomStreamInfo { CustomStreamHandler reader, writer; CustomStreamSeeker seeker; CustomStreamTeller teller; void *data; size_t signature; }; /* Forward declarations. */ static int SyncBlob(Image *); /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + A c q u i r e C u s t o m S t r e a m I n f o % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % AcquireCustomStreamInfo() allocates the CustomStreamInfo structure. % % The format of the AcquireCustomStreamInfo method is: % % CustomStreamInfo *AcquireCustomStreamInfo(ExceptionInfo *exception) % % A description of each parameter follows: % % o exception: return any errors or warnings in this structure. % */ MagickExport CustomStreamInfo *AcquireCustomStreamInfo( ExceptionInfo *magick_unused(exception)) { CustomStreamInfo *custom_stream; magick_unreferenced(exception); custom_stream=(CustomStreamInfo *) AcquireCriticalMemory( sizeof(*custom_stream)); (void) memset(custom_stream,0,sizeof(*custom_stream)); custom_stream->signature=MagickCoreSignature; return(custom_stream); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + A t t a c h B l o b % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % AttachBlob() attaches a blob to the BlobInfo structure. % % The format of the AttachBlob method is: % % void AttachBlob(BlobInfo *blob_info,const void *blob,const size_t length) % % A description of each parameter follows: % % o blob_info: Specifies a pointer to a BlobInfo structure. % % o blob: the address of a character stream in one of the image formats % understood by ImageMagick. % % o length: This size_t integer reflects the length in bytes of the blob. % */ MagickExport void AttachBlob(BlobInfo *blob_info,const void *blob, const size_t length) { assert(blob_info != (BlobInfo *) NULL); if (blob_info->debug != MagickFalse) (void) LogMagickEvent(TraceEvent,GetMagickModule(),"..."); blob_info->length=length; blob_info->extent=length; blob_info->quantum=(size_t) MagickMaxBlobExtent; blob_info->offset=0; blob_info->type=BlobStream; blob_info->file_info.file=(FILE *) NULL; blob_info->data=(unsigned char *) blob; blob_info->mapped=MagickFalse; } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + A t t a c h C u s t o m S t r e a m % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % AttachCustomStream() attaches a CustomStreamInfo to the BlobInfo structure. % % The format of the AttachCustomStream method is: % % void AttachCustomStream(BlobInfo *blob_info, % CustomStreamInfo *custom_stream) % % A description of each parameter follows: % % o blob_info: specifies a pointer to a BlobInfo structure. % % o custom_stream: the custom stream info. % */ MagickExport void AttachCustomStream(BlobInfo *blob_info, CustomStreamInfo *custom_stream) { assert(blob_info != (BlobInfo *) NULL); assert(custom_stream != (CustomStreamInfo *) NULL); assert(custom_stream->signature == MagickCoreSignature); if (blob_info->debug != MagickFalse) (void) LogMagickEvent(TraceEvent,GetMagickModule(),"..."); blob_info->type=CustomStream; blob_info->custom_stream=custom_stream; } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + B l o b T o F i l e % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % BlobToFile() writes a blob to a file. It returns MagickFalse if an error % occurs otherwise MagickTrue. % % The format of the BlobToFile method is: % % MagickBooleanType BlobToFile(char *filename,const void *blob, % const size_t length,ExceptionInfo *exception) % % A description of each parameter follows: % % o filename: Write the blob to this file. % % o blob: the address of a blob. % % o length: This length in bytes of the blob. % % o exception: return any errors or warnings in this structure. % */ MagickExport MagickBooleanType BlobToFile(char *filename,const void *blob, const size_t length,ExceptionInfo *exception) { int file; size_t i; ssize_t count; assert(filename != (const char *) NULL); (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",filename); assert(blob != (const void *) NULL); if (*filename == '\0') file=AcquireUniqueFileResource(filename); else file=open_utf8(filename,O_RDWR | O_CREAT | O_EXCL | O_BINARY,S_MODE); if (file == -1) { ThrowFileException(exception,BlobError,"UnableToWriteBlob",filename); return(MagickFalse); } for (i=0; i < length; i+=count) { count=write(file,(const char *) blob+i,MagickMin(length-i,(size_t) MAGICK_SSIZE_MAX)); if (count <= 0) { count=0; if (errno != EINTR) break; } } file=close(file); if ((file == -1) || (i < length)) { ThrowFileException(exception,BlobError,"UnableToWriteBlob",filename); return(MagickFalse); } return(MagickTrue); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % % B l o b T o I m a g e % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % BlobToImage() implements direct to memory image formats. It returns the % blob as an image. % % The format of the BlobToImage method is: % % Image *BlobToImage(const ImageInfo *image_info,const void *blob, % const size_t length,ExceptionInfo *exception) % % A description of each parameter follows: % % o image_info: the image info. % % o blob: the address of a character stream in one of the image formats % understood by ImageMagick. % % o length: This size_t integer reflects the length in bytes of the blob. % % o exception: return any errors or warnings in this structure. % */ MagickExport Image *BlobToImage(const ImageInfo *image_info,const void *blob, const size_t length,ExceptionInfo *exception) { const MagickInfo *magick_info; Image *image; ImageInfo *blob_info, *clone_info; MagickBooleanType status; assert(image_info != (ImageInfo *) NULL); assert(image_info->signature == MagickCoreSignature); if (image_info->debug != MagickFalse) (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s", image_info->filename); assert(exception != (ExceptionInfo *) NULL); if ((blob == (const void *) NULL) || (length == 0)) { (void) ThrowMagickException(exception,GetMagickModule(),BlobError, "ZeroLengthBlobNotPermitted","`%s'",image_info->filename); return((Image *) NULL); } blob_info=CloneImageInfo(image_info); blob_info->blob=(void *) blob; blob_info->length=length; if (*blob_info->magick == '\0') (void) SetImageInfo(blob_info,0,exception); magick_info=GetMagickInfo(blob_info->magick,exception); if (magick_info == (const MagickInfo *) NULL) { (void) ThrowMagickException(exception,GetMagickModule(), MissingDelegateError,"NoDecodeDelegateForThisImageFormat","`%s'", blob_info->magick); blob_info=DestroyImageInfo(blob_info); return((Image *) NULL); } if (GetMagickBlobSupport(magick_info) != MagickFalse) { char filename[MagickPathExtent]; /* Native blob support for this image format. */ (void) CopyMagickString(filename,blob_info->filename,MagickPathExtent); (void) FormatLocaleString(blob_info->filename,MagickPathExtent,"%s:%s", blob_info->magick,filename); image=ReadImage(blob_info,exception); if (image != (Image *) NULL) (void) DetachBlob(image->blob); blob_info=DestroyImageInfo(blob_info); return(image); } /* Write blob to a temporary file on disk. */ blob_info->blob=(void *) NULL; blob_info->length=0; *blob_info->filename='\0'; status=BlobToFile(blob_info->filename,blob,length,exception); if (status == MagickFalse) { (void) RelinquishUniqueFileResource(blob_info->filename); blob_info=DestroyImageInfo(blob_info); return((Image *) NULL); } clone_info=CloneImageInfo(blob_info); (void) FormatLocaleString(clone_info->filename,MagickPathExtent,"%s:%s", blob_info->magick,blob_info->filename); image=ReadImage(clone_info,exception); if (image != (Image *) NULL) { Image *images; /* Restore original filenames and image format. */ for (images=GetFirstImageInList(image); images != (Image *) NULL; ) { (void) CopyMagickString(images->filename,image_info->filename, MagickPathExtent); (void) CopyMagickString(images->magick_filename,image_info->filename, MagickPathExtent); (void) CopyMagickString(images->magick,magick_info->name, MagickPathExtent); images=GetNextImageInList(images); } } clone_info=DestroyImageInfo(clone_info); (void) RelinquishUniqueFileResource(blob_info->filename); blob_info=DestroyImageInfo(blob_info); return(image); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + C l o n e B l o b I n f o % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % CloneBlobInfo() makes a duplicate of the given blob info structure, or if % blob info is NULL, a new one. % % The format of the CloneBlobInfo method is: % % BlobInfo *CloneBlobInfo(const BlobInfo *blob_info) % % A description of each parameter follows: % % o blob_info: the blob info. % */ MagickExport BlobInfo *CloneBlobInfo(const BlobInfo *blob_info) { BlobInfo *clone_info; SemaphoreInfo *semaphore; clone_info=(BlobInfo *) AcquireCriticalMemory(sizeof(*clone_info)); GetBlobInfo(clone_info); if (blob_info == (BlobInfo *) NULL) return(clone_info); semaphore=clone_info->semaphore; (void) memcpy(clone_info,blob_info,sizeof(*clone_info)); if (blob_info->mapped != MagickFalse) (void) AcquireMagickResource(MapResource,blob_info->length); clone_info->semaphore=semaphore; LockSemaphoreInfo(clone_info->semaphore); clone_info->reference_count=1; UnlockSemaphoreInfo(clone_info->semaphore); return(clone_info); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + C l o s e B l o b % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % CloseBlob() closes a stream associated with the image. % % The format of the CloseBlob method is: % % MagickBooleanType CloseBlob(Image *image) % % A description of each parameter follows: % % o image: the image. % */ static inline void ThrowBlobException(BlobInfo *blob_info) { if ((blob_info->status == MagickFalse) && (errno != 0)) blob_info->error_number=errno; blob_info->status=MagickTrue; } MagickExport MagickBooleanType CloseBlob(Image *image) { BlobInfo *magick_restrict blob_info; int status; /* Close image file. */ assert(image != (Image *) NULL); assert(image->signature == MagickCoreSignature); if (image->debug != MagickFalse) (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",image->filename); blob_info=image->blob; if ((blob_info == (BlobInfo *) NULL) || (blob_info->type == UndefinedStream)) return(MagickTrue); status=SyncBlob(image); switch (blob_info->type) { case UndefinedStream: case StandardStream: break; case FileStream: case PipeStream: { if (blob_info->synchronize != MagickFalse) { status=fflush(blob_info->file_info.file); if (status != 0) ThrowBlobException(blob_info); status=fsync(fileno(blob_info->file_info.file)); if (status != 0) ThrowBlobException(blob_info); } if ((status != 0) && (ferror(blob_info->file_info.file) != 0)) ThrowBlobException(blob_info); break; } case ZipStream: { #if defined(MAGICKCORE_ZLIB_DELEGATE) status=Z_OK; (void) gzerror(blob_info->file_info.gzfile,&status); if (status != Z_OK) ThrowBlobException(blob_info); #endif break; } case BZipStream: { #if defined(MAGICKCORE_BZLIB_DELEGATE) status=BZ_OK; (void) BZ2_bzerror(blob_info->file_info.bzfile,&status); if (status != BZ_OK) ThrowBlobException(blob_info); #endif break; } case FifoStream: break; case BlobStream: { if (blob_info->file_info.file != (FILE *) NULL) { if (blob_info->synchronize != MagickFalse) { status=fflush(blob_info->file_info.file); if (status != 0) ThrowBlobException(blob_info); status=fsync(fileno(blob_info->file_info.file)); if (status != 0) ThrowBlobException(blob_info); } if ((status != 0) && (ferror(blob_info->file_info.file) != 0)) ThrowBlobException(blob_info); } break; } case CustomStream: break; } blob_info->size=GetBlobSize(image); image->extent=blob_info->size; blob_info->eof=MagickFalse; blob_info->error=0; blob_info->mode=UndefinedBlobMode; if (blob_info->exempt != MagickFalse) { blob_info->type=UndefinedStream; return(blob_info->status); } switch (blob_info->type) { case UndefinedStream: case StandardStream: break; case FileStream: { if (fileno(blob_info->file_info.file) != -1) { status=fclose(blob_info->file_info.file); if (status != 0) ThrowBlobException(blob_info); } break; } case PipeStream: { #if defined(MAGICKCORE_HAVE_PCLOSE) status=pclose(blob_info->file_info.file); if (status != 0) ThrowBlobException(blob_info); #endif break; } case ZipStream: { #if defined(MAGICKCORE_ZLIB_DELEGATE) status=gzclose(blob_info->file_info.gzfile); if (status != Z_OK) ThrowBlobException(blob_info); #endif break; } case BZipStream: { #if defined(MAGICKCORE_BZLIB_DELEGATE) BZ2_bzclose(blob_info->file_info.bzfile); #endif break; } case FifoStream: break; case BlobStream: { if (blob_info->file_info.file != (FILE *) NULL) { status=fclose(blob_info->file_info.file); if (status != 0) ThrowBlobException(blob_info); } break; } case CustomStream: break; } (void) DetachBlob(blob_info); return(blob_info->status); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % % C u s t o m S t r e a m T o I m a g e % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % CustomStreamToImage() is the equivalent of ReadImage(), but reads the % formatted "file" from the suplied method rather than to an actual file. % % The format of the CustomStreamToImage method is: % % Image *CustomStreamToImage(const ImageInfo *image_info, % ExceptionInfo *exception) % % A description of each parameter follows: % % o image_info: the image info. % % o exception: return any errors or warnings in this structure. % */ MagickExport Image *CustomStreamToImage(const ImageInfo *image_info, ExceptionInfo *exception) { const MagickInfo *magick_info; Image *image; ImageInfo *blob_info; assert(image_info != (ImageInfo *) NULL); assert(image_info->signature == MagickCoreSignature); if (image_info->debug != MagickFalse) (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s", image_info->filename); assert(image_info->custom_stream != (CustomStreamInfo *) NULL); assert(image_info->custom_stream->signature == MagickCoreSignature); assert(image_info->custom_stream->reader != (CustomStreamHandler) NULL); assert(exception != (ExceptionInfo *) NULL); blob_info=CloneImageInfo(image_info); if (*blob_info->magick == '\0') (void) SetImageInfo(blob_info,0,exception); magick_info=GetMagickInfo(blob_info->magick,exception); if (magick_info == (const MagickInfo *) NULL) { (void) ThrowMagickException(exception,GetMagickModule(), MissingDelegateError,"NoDecodeDelegateForThisImageFormat","`%s'", blob_info->magick); blob_info=DestroyImageInfo(blob_info); return((Image *) NULL); } image=(Image *) NULL; if ((GetMagickBlobSupport(magick_info) != MagickFalse) || (*blob_info->filename != '\0')) { char filename[MagickPathExtent]; /* Native blob support for this image format or SetImageInfo changed the blob to a file. */ (void) CopyMagickString(filename,blob_info->filename,MagickPathExtent); (void) FormatLocaleString(blob_info->filename,MagickPathExtent,"%s:%s", blob_info->magick,filename); image=ReadImage(blob_info,exception); if (image != (Image *) NULL) (void) CloseBlob(image); } else { char unique[MagickPathExtent]; int file; ImageInfo *clone_info; unsigned char *blob; /* Write data to file on disk. */ blob_info->custom_stream=(CustomStreamInfo *) NULL; blob=(unsigned char *) AcquireQuantumMemory(MagickMaxBufferExtent, sizeof(*blob)); if (blob == (unsigned char *) NULL) { ThrowFileException(exception,BlobError,"UnableToReadBlob", image_info->filename); blob_info=DestroyImageInfo(blob_info); return((Image *) NULL); } file=AcquireUniqueFileResource(unique); if (file == -1) { ThrowFileException(exception,BlobError,"UnableToReadBlob", image_info->filename); blob=(unsigned char *) RelinquishMagickMemory(blob); blob_info=DestroyImageInfo(blob_info); return((Image *) NULL); } clone_info=CloneImageInfo(blob_info); blob_info->file=fdopen(file,"wb+"); if (blob_info->file != (FILE *) NULL) { ssize_t count; count=(ssize_t) MagickMaxBufferExtent; while (count == (ssize_t) MagickMaxBufferExtent) { count=image_info->custom_stream->reader(blob,MagickMaxBufferExtent, image_info->custom_stream->data); count=(ssize_t) write(file,(const char *) blob,(size_t) count); } (void) fclose(blob_info->file); (void) FormatLocaleString(clone_info->filename,MagickPathExtent, "%s:%s",blob_info->magick,unique); image=ReadImage(clone_info,exception); if (image != (Image *) NULL) { Image *images; /* Restore original filenames and image format. */ for (images=GetFirstImageInList(image); images != (Image *) NULL; ) { (void) CopyMagickString(images->filename,image_info->filename, MagickPathExtent); (void) CopyMagickString(images->magick_filename, image_info->filename,MagickPathExtent); (void) CopyMagickString(images->magick,magick_info->name, MagickPathExtent); (void) CloseBlob(images); images=GetNextImageInList(images); } } } clone_info=DestroyImageInfo(clone_info); blob=(unsigned char *) RelinquishMagickMemory(blob); (void) RelinquishUniqueFileResource(unique); } blob_info=DestroyImageInfo(blob_info); return(image); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + D e s t r o y B l o b % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % DestroyBlob() deallocates memory associated with a blob. % % The format of the DestroyBlob method is: % % void DestroyBlob(Image *image) % % A description of each parameter follows: % % o image: the image. % */ MagickExport void DestroyBlob(Image *image) { BlobInfo *magick_restrict blob_info; MagickBooleanType destroy; assert(image != (Image *) NULL); assert(image->signature == MagickCoreSignature); if (image->debug != MagickFalse) (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",image->filename); assert(image->blob != (BlobInfo *) NULL); assert(image->blob->signature == MagickCoreSignature); blob_info=image->blob; destroy=MagickFalse; LockSemaphoreInfo(blob_info->semaphore); blob_info->reference_count--; assert(blob_info->reference_count >= 0); if (blob_info->reference_count == 0) destroy=MagickTrue; UnlockSemaphoreInfo(blob_info->semaphore); if (destroy == MagickFalse) { image->blob=(BlobInfo *) NULL; return; } (void) CloseBlob(image); if (blob_info->mapped != MagickFalse) { (void) UnmapBlob(blob_info->data,blob_info->length); RelinquishMagickResource(MapResource,blob_info->length); } if (blob_info->semaphore != (SemaphoreInfo *) NULL) RelinquishSemaphoreInfo(&blob_info->semaphore); blob_info->signature=(~MagickCoreSignature); image->blob=(BlobInfo *) RelinquishMagickMemory(blob_info); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + D e s t r o y C u s t o m S t r e a m I n f o % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % DestroyCustomStreamInfo() destroys memory associated with the % CustomStreamInfo structure. % % The format of the DestroyCustomStreamInfo method is: % % CustomStreamInfo *DestroyCustomStreamInfo(CustomStreamInfo *stream_info) % % A description of each parameter follows: % % o custom_stream: the custom stream info. % */ MagickExport CustomStreamInfo *DestroyCustomStreamInfo( CustomStreamInfo *custom_stream) { (void) LogMagickEvent(TraceEvent,GetMagickModule(),"..."); assert(custom_stream != (CustomStreamInfo *) NULL); assert(custom_stream->signature == MagickCoreSignature); custom_stream->signature=(~MagickCoreSignature); custom_stream=(CustomStreamInfo *) RelinquishMagickMemory(custom_stream); return(custom_stream); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + D e t a c h B l o b % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % DetachBlob() detaches a blob from the BlobInfo structure. % % The format of the DetachBlob method is: % % void *DetachBlob(BlobInfo *blob_info) % % A description of each parameter follows: % % o blob_info: Specifies a pointer to a BlobInfo structure. % */ MagickExport void *DetachBlob(BlobInfo *blob_info) { void *data; assert(blob_info != (BlobInfo *) NULL); if (blob_info->debug != MagickFalse) (void) LogMagickEvent(TraceEvent,GetMagickModule(),"..."); if (blob_info->mapped != MagickFalse) { (void) UnmapBlob(blob_info->data,blob_info->length); blob_info->data=NULL; RelinquishMagickResource(MapResource,blob_info->length); } blob_info->mapped=MagickFalse; blob_info->length=0; blob_info->offset=0; blob_info->eof=MagickFalse; blob_info->error=0; blob_info->exempt=MagickFalse; blob_info->type=UndefinedStream; blob_info->file_info.file=(FILE *) NULL; data=blob_info->data; blob_info->data=(unsigned char *) NULL; blob_info->stream=(StreamHandler) NULL; blob_info->custom_stream=(CustomStreamInfo *) NULL; return(data); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + D i s a s s o c i a t e B l o b % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % DisassociateBlob() disassociates the image stream. It checks if the % blob of the specified image is referenced by other images. If the reference % count is higher then 1 a new blob is assigned to the specified image. % % The format of the DisassociateBlob method is: % % void DisassociateBlob(const Image *image) % % A description of each parameter follows: % % o image: the image. % */ MagickExport void DisassociateBlob(Image *image) { BlobInfo *magick_restrict blob_info, *clone_info; MagickBooleanType clone; assert(image != (Image *) NULL); assert(image->signature == MagickCoreSignature); if (image->debug != MagickFalse) (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",image->filename); assert(image->blob != (BlobInfo *) NULL); assert(image->blob->signature == MagickCoreSignature); blob_info=image->blob; clone=MagickFalse; LockSemaphoreInfo(blob_info->semaphore); assert(blob_info->reference_count >= 0); if (blob_info->reference_count > 1) clone=MagickTrue; UnlockSemaphoreInfo(blob_info->semaphore); if (clone == MagickFalse) return; clone_info=CloneBlobInfo(blob_info); DestroyBlob(image); image->blob=clone_info; } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + D i s c a r d B l o b B y t e s % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % DiscardBlobBytes() discards bytes in a blob. % % The format of the DiscardBlobBytes method is: % % MagickBooleanType DiscardBlobBytes(Image *image, % const MagickSizeType length) % % A description of each parameter follows. % % o image: the image. % % o length: the number of bytes to skip. % */ MagickExport MagickBooleanType DiscardBlobBytes(Image *image, const MagickSizeType length) { MagickOffsetType i; size_t quantum; ssize_t count; unsigned char buffer[MagickMinBufferExtent >> 1]; assert(image != (Image *) NULL); assert(image->signature == MagickCoreSignature); if (length != (MagickSizeType) ((MagickOffsetType) length)) return(MagickFalse); count=0; for (i=0; i < (MagickOffsetType) length; i+=count) { quantum=(size_t) MagickMin(length-i,sizeof(buffer)); (void) ReadBlobStream(image,quantum,buffer,&count); if (count <= 0) { count=0; if (errno != EINTR) break; } } return(i < (MagickOffsetType) length ? MagickFalse : MagickTrue); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + D u p l i c a t e s B l o b % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % DuplicateBlob() duplicates a blob descriptor. % % The format of the DuplicateBlob method is: % % void DuplicateBlob(Image *image,const Image *duplicate) % % A description of each parameter follows: % % o image: the image. % % o duplicate: the duplicate image. % */ MagickExport void DuplicateBlob(Image *image,const Image *duplicate) { assert(image != (Image *) NULL); assert(image->signature == MagickCoreSignature); if (image->debug != MagickFalse) (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",image->filename); assert(duplicate != (Image *) NULL); assert(duplicate->signature == MagickCoreSignature); DestroyBlob(image); image->blob=ReferenceBlob(duplicate->blob); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + E O F B l o b % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % EOFBlob() returns a non-zero value when EOF has been detected reading from % a blob or file. % % The format of the EOFBlob method is: % % int EOFBlob(const Image *image) % % A description of each parameter follows: % % o image: the image. % */ MagickExport int EOFBlob(const Image *image) { BlobInfo *magick_restrict blob_info; assert(image != (Image *) NULL); assert(image->signature == MagickCoreSignature); if (image->debug != MagickFalse) (void) LogMagickEvent(TraceEvent,GetMagickModule(),"..."); assert(image->blob != (BlobInfo *) NULL); assert(image->blob->type != UndefinedStream); blob_info=image->blob; switch (blob_info->type) { case UndefinedStream: case StandardStream: break; case FileStream: case PipeStream: { blob_info->eof=feof(blob_info->file_info.file) != 0 ? MagickTrue : MagickFalse; break; } case ZipStream: { #if defined(MAGICKCORE_ZLIB_DELEGATE) blob_info->eof=gzeof(blob_info->file_info.gzfile) != 0 ? MagickTrue : MagickFalse; #endif break; } case BZipStream: { #if defined(MAGICKCORE_BZLIB_DELEGATE) int status; status=0; (void) BZ2_bzerror(blob_info->file_info.bzfile,&status); blob_info->eof=status == BZ_UNEXPECTED_EOF ? MagickTrue : MagickFalse; #endif break; } case FifoStream: { blob_info->eof=MagickFalse; break; } case BlobStream: break; case CustomStream: break; } return((int) blob_info->eof); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + E r r o r B l o b % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % ErrorBlob() returns a non-zero value when an error has been detected reading % from a blob or file. % % The format of the ErrorBlob method is: % % int ErrorBlob(const Image *image) % % A description of each parameter follows: % % o image: the image. % */ MagickExport int ErrorBlob(const Image *image) { BlobInfo *magick_restrict blob_info; assert(image != (Image *) NULL); assert(image->signature == MagickCoreSignature); if (image->debug != MagickFalse) (void) LogMagickEvent(TraceEvent,GetMagickModule(),"..."); assert(image->blob != (BlobInfo *) NULL); assert(image->blob->type != UndefinedStream); blob_info=image->blob; switch (blob_info->type) { case UndefinedStream: case StandardStream: break; case FileStream: case PipeStream: { blob_info->error=ferror(blob_info->file_info.file); break; } case ZipStream: { #if defined(MAGICKCORE_ZLIB_DELEGATE) (void) gzerror(blob_info->file_info.gzfile,&blob_info->error); #endif break; } case BZipStream: { #if defined(MAGICKCORE_BZLIB_DELEGATE) (void) BZ2_bzerror(blob_info->file_info.bzfile,&blob_info->error); #endif break; } case FifoStream: { blob_info->error=0; break; } case BlobStream: break; case CustomStream: break; } return(blob_info->error); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % % F i l e T o B l o b % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % FileToBlob() returns the contents of a file as a buffer terminated with % the '\0' character. The length of the buffer (not including the extra % terminating '\0' character) is returned via the 'length' parameter. Free % the buffer with RelinquishMagickMemory(). % % The format of the FileToBlob method is: % % void *FileToBlob(const char *filename,const size_t extent, % size_t *length,ExceptionInfo *exception) % % A description of each parameter follows: % % o blob: FileToBlob() returns the contents of a file as a blob. If % an error occurs NULL is returned. % % o filename: the filename. % % o extent: The maximum length of the blob. % % o length: On return, this reflects the actual length of the blob. % % o exception: return any errors or warnings in this structure. % */ MagickExport void *FileToBlob(const char *filename,const size_t extent, size_t *length,ExceptionInfo *exception) { int file; MagickBooleanType status; MagickOffsetType offset; size_t i; ssize_t count; struct stat attributes; unsigned char *blob; void *map; assert(filename != (const char *) NULL); (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",filename); assert(exception != (ExceptionInfo *) NULL); *length=0; status=IsRightsAuthorized(PathPolicyDomain,ReadPolicyRights,filename); if (status == MagickFalse) { errno=EPERM; (void) ThrowMagickException(exception,GetMagickModule(),PolicyError, "NotAuthorized","`%s'",filename); return(NULL); } file=fileno(stdin); if (LocaleCompare(filename,"-") != 0) { status=GetPathAttributes(filename,&attributes); if ((status == MagickFalse) || (S_ISDIR(attributes.st_mode) != 0)) { ThrowFileException(exception,BlobError,"UnableToReadBlob",filename); return(NULL); } file=open_utf8(filename,O_RDONLY | O_BINARY,0); } if (file == -1) { ThrowFileException(exception,BlobError,"UnableToOpenFile",filename); return(NULL); } offset=(MagickOffsetType) lseek(file,0,SEEK_END); count=0; if ((file == fileno(stdin)) || (offset < 0) || (offset != (MagickOffsetType) ((ssize_t) offset))) { size_t quantum; struct stat file_stats; /* Stream is not seekable. */ offset=(MagickOffsetType) lseek(file,0,SEEK_SET); quantum=(size_t) MagickMaxBufferExtent; if ((fstat(file,&file_stats) == 0) && (file_stats.st_size > 0)) quantum=(size_t) MagickMin(file_stats.st_size,MagickMaxBufferExtent); blob=(unsigned char *) AcquireQuantumMemory(quantum,sizeof(*blob)); for (i=0; blob != (unsigned char *) NULL; i+=count) { count=read(file,blob+i,quantum); if (count <= 0) { count=0; if (errno != EINTR) break; } if (~((size_t) i) < (quantum+1)) { blob=(unsigned char *) RelinquishMagickMemory(blob); break; } blob=(unsigned char *) ResizeQuantumMemory(blob,i+quantum+1, sizeof(*blob)); if ((size_t) (i+count) >= extent) break; } if (LocaleCompare(filename,"-") != 0) file=close(file); if (blob == (unsigned char *) NULL) { (void) ThrowMagickException(exception,GetMagickModule(), ResourceLimitError,"MemoryAllocationFailed","`%s'",filename); return(NULL); } if (file == -1) { blob=(unsigned char *) RelinquishMagickMemory(blob); ThrowFileException(exception,BlobError,"UnableToReadBlob",filename); return(NULL); } *length=(size_t) MagickMin(i+count,extent); blob[*length]='\0'; return(blob); } *length=(size_t) MagickMin(offset,(MagickOffsetType) MagickMin(extent,(size_t) MAGICK_SSIZE_MAX)); blob=(unsigned char *) NULL; if (~(*length) >= (MagickPathExtent-1)) blob=(unsigned char *) AcquireQuantumMemory(*length+MagickPathExtent, sizeof(*blob)); if (blob == (unsigned char *) NULL) { file=close(file); (void) ThrowMagickException(exception,GetMagickModule(), ResourceLimitError,"MemoryAllocationFailed","`%s'",filename); return(NULL); } map=MapBlob(file,ReadMode,0,*length); if (map != (unsigned char *) NULL) { (void) memcpy(blob,map,*length); (void) UnmapBlob(map,*length); } else { (void) lseek(file,0,SEEK_SET); for (i=0; i < *length; i+=count) { count=read(file,blob+i,(size_t) MagickMin(*length-i,(size_t) MAGICK_SSIZE_MAX)); if (count <= 0) { count=0; if (errno != EINTR) break; } } if (i < *length) { file=close(file)-1; blob=(unsigned char *) RelinquishMagickMemory(blob); ThrowFileException(exception,BlobError,"UnableToReadBlob",filename); return(NULL); } } blob[*length]='\0'; if (LocaleCompare(filename,"-") != 0) file=close(file); if (file == -1) { blob=(unsigned char *) RelinquishMagickMemory(blob); ThrowFileException(exception,BlobError,"UnableToReadBlob",filename); } return(blob); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % % F i l e T o I m a g e % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % FileToImage() write the contents of a file to an image. % % The format of the FileToImage method is: % % MagickBooleanType FileToImage(Image *,const char *filename) % % A description of each parameter follows: % % o image: the image. % % o filename: the filename. % */ static inline ssize_t WriteBlobStream(Image *image,const size_t length, const void *magick_restrict data) { BlobInfo *magick_restrict blob_info; MagickSizeType extent; unsigned char *magick_restrict q; assert(image->blob != (BlobInfo *) NULL); assert(image->blob->type != UndefinedStream); assert(data != NULL); blob_info=image->blob; if (blob_info->type != BlobStream) return(WriteBlob(image,length,(const unsigned char *) data)); extent=(MagickSizeType) (blob_info->offset+(MagickOffsetType) length); if (extent >= blob_info->extent) { extent=blob_info->extent+blob_info->quantum+length; blob_info->quantum<<=1; if (SetBlobExtent(image,extent) == MagickFalse) return(0); } q=blob_info->data+blob_info->offset; (void) memcpy(q,data,length); blob_info->offset+=length; if (blob_info->offset >= (MagickOffsetType) blob_info->length) blob_info->length=(size_t) blob_info->offset; return((ssize_t) length); } MagickExport MagickBooleanType FileToImage(Image *image,const char *filename, ExceptionInfo *exception) { int file; MagickBooleanType status; size_t length, quantum; ssize_t count; struct stat file_stats; unsigned char *blob; assert(image != (const Image *) NULL); assert(image->signature == MagickCoreSignature); assert(filename != (const char *) NULL); (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",filename); status=IsRightsAuthorized(PathPolicyDomain,WritePolicyRights,filename); if (status == MagickFalse) { errno=EPERM; (void) ThrowMagickException(exception,GetMagickModule(),PolicyError, "NotAuthorized","`%s'",filename); return(MagickFalse); } file=fileno(stdin); if (LocaleCompare(filename,"-") != 0) file=open_utf8(filename,O_RDONLY | O_BINARY,0); if (file == -1) { ThrowFileException(exception,BlobError,"UnableToOpenBlob",filename); return(MagickFalse); } quantum=(size_t) MagickMaxBufferExtent; if ((fstat(file,&file_stats) == 0) && (file_stats.st_size > 0)) quantum=(size_t) MagickMin(file_stats.st_size,MagickMaxBufferExtent); blob=(unsigned char *) AcquireQuantumMemory(quantum,sizeof(*blob)); if (blob == (unsigned char *) NULL) { file=close(file); ThrowFileException(exception,ResourceLimitError,"MemoryAllocationFailed", filename); return(MagickFalse); } for ( ; ; ) { count=read(file,blob,quantum); if (count <= 0) { count=0; if (errno != EINTR) break; } length=(size_t) count; count=WriteBlobStream(image,length,blob); if (count != (ssize_t) length) { ThrowFileException(exception,BlobError,"UnableToWriteBlob",filename); break; } } file=close(file); if (file == -1) ThrowFileException(exception,BlobError,"UnableToWriteBlob",filename); blob=(unsigned char *) RelinquishMagickMemory(blob); return(MagickTrue); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + G e t B l o b E r r o r % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % GetBlobError() returns MagickTrue if the blob associated with the specified % image encountered an error. % % The format of the GetBlobError method is: % % MagickBooleanType GetBlobError(const Image *image) % % A description of each parameter follows: % % o image: the image. % */ MagickExport MagickBooleanType GetBlobError(const Image *image) { assert(image != (const Image *) NULL); assert(image->signature == MagickCoreSignature); if (image->debug != MagickFalse) (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",image->filename); if ((image->blob->status != MagickFalse) && (image->blob->error_number != 0)) errno=image->blob->error_number; return(image->blob->status); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + G e t B l o b F i l e H a n d l e % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % GetBlobFileHandle() returns the file handle associated with the image blob. % % The format of the GetBlobFile method is: % % FILE *GetBlobFileHandle(const Image *image) % % A description of each parameter follows: % % o image: the image. % */ MagickExport FILE *GetBlobFileHandle(const Image *image) { assert(image != (const Image *) NULL); assert(image->signature == MagickCoreSignature); return(image->blob->file_info.file); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + G e t B l o b I n f o % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % GetBlobInfo() initializes the BlobInfo structure. % % The format of the GetBlobInfo method is: % % void GetBlobInfo(BlobInfo *blob_info) % % A description of each parameter follows: % % o blob_info: Specifies a pointer to a BlobInfo structure. % */ MagickExport void GetBlobInfo(BlobInfo *blob_info) { assert(blob_info != (BlobInfo *) NULL); (void) memset(blob_info,0,sizeof(*blob_info)); blob_info->type=UndefinedStream; blob_info->quantum=(size_t) MagickMaxBlobExtent; blob_info->properties.st_mtime=GetMagickTime(); blob_info->properties.st_ctime=blob_info->properties.st_mtime; blob_info->debug=IsEventLogging(); blob_info->reference_count=1; blob_info->semaphore=AcquireSemaphoreInfo(); blob_info->signature=MagickCoreSignature; } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % % G e t B l o b P r o p e r t i e s % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % GetBlobProperties() returns information about an image blob. % % The format of the GetBlobProperties method is: % % const struct stat *GetBlobProperties(const Image *image) % % A description of each parameter follows: % % o image: the image. % */ MagickExport const struct stat *GetBlobProperties(const Image *image) { assert(image != (Image *) NULL); assert(image->signature == MagickCoreSignature); if (image->debug != MagickFalse) (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",image->filename); return(&image->blob->properties); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + G e t B l o b S i z e % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % GetBlobSize() returns the current length of the image file or blob; zero is % returned if the size cannot be determined. % % The format of the GetBlobSize method is: % % MagickSizeType GetBlobSize(const Image *image) % % A description of each parameter follows: % % o image: the image. % */ MagickExport MagickSizeType GetBlobSize(const Image *image) { BlobInfo *magick_restrict blob_info; MagickSizeType extent; assert(image != (Image *) NULL); assert(image->signature == MagickCoreSignature); if (image->debug != MagickFalse) (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",image->filename); assert(image->blob != (BlobInfo *) NULL); blob_info=image->blob; extent=0; switch (blob_info->type) { case UndefinedStream: case StandardStream: { extent=blob_info->size; break; } case FileStream: { int file_descriptor; extent=(MagickSizeType) blob_info->properties.st_size; if (extent == 0) extent=blob_info->size; file_descriptor=fileno(blob_info->file_info.file); if (file_descriptor == -1) break; if (fstat(file_descriptor,&blob_info->properties) == 0) extent=(MagickSizeType) blob_info->properties.st_size; break; } case PipeStream: { extent=blob_info->size; break; } case ZipStream: case BZipStream: { MagickBooleanType status; status=GetPathAttributes(image->filename,&blob_info->properties); if (status != MagickFalse) extent=(MagickSizeType) blob_info->properties.st_size; break; } case FifoStream: break; case BlobStream: { extent=(MagickSizeType) blob_info->length; break; } case CustomStream: { if ((blob_info->custom_stream->teller != (CustomStreamTeller) NULL) && (blob_info->custom_stream->seeker != (CustomStreamSeeker) NULL)) { MagickOffsetType offset; offset=blob_info->custom_stream->teller( blob_info->custom_stream->data); extent=(MagickSizeType) blob_info->custom_stream->seeker(0,SEEK_END, blob_info->custom_stream->data); (void) blob_info->custom_stream->seeker(offset,SEEK_SET, blob_info->custom_stream->data); } break; } } return(extent); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + G e t B l o b S t r e a m D a t a % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % GetBlobStreamData() returns the stream data for the image. % % The format of the GetBlobStreamData method is: % % void *GetBlobStreamData(const Image *image) % % A description of each parameter follows: % % o image: the image. % */ MagickExport void *GetBlobStreamData(const Image *image) { assert(image != (const Image *) NULL); assert(image->signature == MagickCoreSignature); return(image->blob->data); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + G e t B l o b S t r e a m H a n d l e r % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % GetBlobStreamHandler() returns the stream handler for the image. % % The format of the GetBlobStreamHandler method is: % % StreamHandler GetBlobStreamHandler(const Image *image) % % A description of each parameter follows: % % o image: the image. % */ MagickExport StreamHandler GetBlobStreamHandler(const Image *image) { assert(image != (const Image *) NULL); assert(image->signature == MagickCoreSignature); if (image->debug != MagickFalse) (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",image->filename); return(image->blob->stream); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % % I m a g e T o B l o b % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % ImageToBlob() implements direct to memory image formats. It returns the % image as a formatted blob and its length. The magick member of the Image % structure determines the format of the returned blob (GIF, JPEG, PNG, % etc.). This method is the equivalent of WriteImage(), but writes the % formatted "file" to a memory buffer rather than to an actual file. % % The format of the ImageToBlob method is: % % void *ImageToBlob(const ImageInfo *image_info,Image *image, % size_t *length,ExceptionInfo *exception) % % A description of each parameter follows: % % o image_info: the image info. % % o image: the image. % % o length: return the actual length of the blob. % % o exception: return any errors or warnings in this structure. % */ MagickExport void *ImageToBlob(const ImageInfo *image_info, Image *image,size_t *length,ExceptionInfo *exception) { const MagickInfo *magick_info; ImageInfo *blob_info; MagickBooleanType status; void *blob; assert(image_info != (const ImageInfo *) NULL); assert(image_info->signature == MagickCoreSignature); if (image_info->debug != MagickFalse) (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s", image_info->filename); assert(image != (Image *) NULL); assert(image->signature == MagickCoreSignature); assert(exception != (ExceptionInfo *) NULL); *length=0; blob=(unsigned char *) NULL; blob_info=CloneImageInfo(image_info); blob_info->adjoin=MagickFalse; (void) SetImageInfo(blob_info,1,exception); if (*blob_info->magick != '\0') (void) CopyMagickString(image->magick,blob_info->magick,MagickPathExtent); magick_info=GetMagickInfo(image->magick,exception); if (magick_info == (const MagickInfo *) NULL) { (void) ThrowMagickException(exception,GetMagickModule(), MissingDelegateError,"NoDecodeDelegateForThisImageFormat","`%s'", image->magick); blob_info=DestroyImageInfo(blob_info); return(blob); } (void) CopyMagickString(blob_info->magick,image->magick,MagickPathExtent); if (GetMagickBlobSupport(magick_info) != MagickFalse) { /* Native blob support for this image format. */ blob_info->length=0; blob_info->blob=AcquireQuantumMemory(MagickMaxBlobExtent, sizeof(unsigned char)); if (blob_info->blob == NULL) (void) ThrowMagickException(exception,GetMagickModule(), ResourceLimitError,"MemoryAllocationFailed","`%s'",image->filename); else { (void) CloseBlob(image); image->blob->exempt=MagickTrue; *image->filename='\0'; status=WriteImage(blob_info,image,exception); *length=image->blob->length; blob=DetachBlob(image->blob); if (blob == (void *) NULL) blob_info->blob=RelinquishMagickMemory(blob_info->blob); else if (status == MagickFalse) blob=RelinquishMagickMemory(blob); else blob=ResizeQuantumMemory(blob,*length+1,sizeof(unsigned char)); } } else { char unique[MagickPathExtent]; int file; /* Write file to disk in blob image format. */ file=AcquireUniqueFileResource(unique); if (file == -1) { ThrowFileException(exception,BlobError,"UnableToWriteBlob", image_info->filename); } else { blob_info->file=fdopen(file,"wb"); if (blob_info->file != (FILE *) NULL) { (void) FormatLocaleString(image->filename,MagickPathExtent, "%s:%s",image->magick,unique); status=WriteImage(blob_info,image,exception); (void) CloseBlob(image); (void) fclose(blob_info->file); if (status != MagickFalse) blob=FileToBlob(unique,~0UL,length,exception); } (void) RelinquishUniqueFileResource(unique); } } blob_info=DestroyImageInfo(blob_info); return(blob); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + I m a g e T o C u s t o m S t r e a m % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % ImageToCustomStream() is the equivalent of WriteImage(), but writes the % formatted "file" to the custom stream rather than to an actual file. % % The format of the ImageToCustomStream method is: % % void ImageToCustomStream(const ImageInfo *image_info,Image *image, % ExceptionInfo *exception) % % A description of each parameter follows: % % o image_info: the image info. % % o image: the image. % % o exception: return any errors or warnings in this structure. % */ MagickExport void ImageToCustomStream(const ImageInfo *image_info,Image *image, ExceptionInfo *exception) { const MagickInfo *magick_info; ImageInfo *clone_info; MagickBooleanType blob_support, status; assert(image_info != (const ImageInfo *) NULL); assert(image_info->signature == MagickCoreSignature); if (image_info->debug != MagickFalse) (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s", image_info->filename); assert(image != (Image *) NULL); assert(image->signature == MagickCoreSignature); assert(image_info->custom_stream != (CustomStreamInfo *) NULL); assert(image_info->custom_stream->signature == MagickCoreSignature); assert(image_info->custom_stream->writer != (CustomStreamHandler) NULL); assert(exception != (ExceptionInfo *) NULL); clone_info=CloneImageInfo(image_info); clone_info->adjoin=MagickFalse; (void) SetImageInfo(clone_info,1,exception); if (*clone_info->magick != '\0') (void) CopyMagickString(image->magick,clone_info->magick,MagickPathExtent); magick_info=GetMagickInfo(image->magick,exception); if (magick_info == (const MagickInfo *) NULL) { (void) ThrowMagickException(exception,GetMagickModule(), MissingDelegateError,"NoEncodeDelegateForThisImageFormat","`%s'", image->magick); clone_info=DestroyImageInfo(clone_info); return; } (void) CopyMagickString(clone_info->magick,image->magick,MagickPathExtent); blob_support=GetMagickBlobSupport(magick_info); if ((blob_support != MagickFalse) && (GetMagickEncoderSeekableStream(magick_info) != MagickFalse)) { if ((clone_info->custom_stream->seeker == (CustomStreamSeeker) NULL) || (clone_info->custom_stream->teller == (CustomStreamTeller) NULL)) blob_support=MagickFalse; } if (blob_support != MagickFalse) { /* Native blob support for this image format. */ (void) CloseBlob(image); *image->filename='\0'; (void) WriteImage(clone_info,image,exception); (void) CloseBlob(image); } else { char unique[MagickPathExtent]; int file; unsigned char *blob; /* Write file to disk in blob image format. */ clone_info->custom_stream=(CustomStreamInfo *) NULL; blob=(unsigned char *) AcquireQuantumMemory(MagickMaxBufferExtent, sizeof(*blob)); if (blob == (unsigned char *) NULL) { ThrowFileException(exception,BlobError,"UnableToWriteBlob", image_info->filename); clone_info=DestroyImageInfo(clone_info); return; } file=AcquireUniqueFileResource(unique); if (file == -1) { ThrowFileException(exception,BlobError,"UnableToWriteBlob", image_info->filename); blob=(unsigned char *) RelinquishMagickMemory(blob); clone_info=DestroyImageInfo(clone_info); return; } clone_info->file=fdopen(file,"wb+"); if (clone_info->file != (FILE *) NULL) { ssize_t count; (void) FormatLocaleString(image->filename,MagickPathExtent, "%s:%s",image->magick,unique); status=WriteImage(clone_info,image,exception); (void) CloseBlob(image); if (status != MagickFalse) { (void) fseek(clone_info->file,0,SEEK_SET); count=(ssize_t) MagickMaxBufferExtent; while (count == (ssize_t) MagickMaxBufferExtent) { count=(ssize_t) fread(blob,sizeof(*blob),MagickMaxBufferExtent, clone_info->file); (void) image_info->custom_stream->writer(blob,(size_t) count, image_info->custom_stream->data); } } (void) fclose(clone_info->file); } blob=(unsigned char *) RelinquishMagickMemory(blob); (void) RelinquishUniqueFileResource(unique); } clone_info=DestroyImageInfo(clone_info); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % % I m a g e T o F i l e % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % ImageToFile() writes an image to a file. It returns MagickFalse if an error % occurs otherwise MagickTrue. % % The format of the ImageToFile method is: % % MagickBooleanType ImageToFile(Image *image,char *filename, % ExceptionInfo *exception) % % A description of each parameter follows: % % o image: the image. % % o filename: Write the image to this file. % % o exception: return any errors or warnings in this structure. % */ MagickExport MagickBooleanType ImageToFile(Image *image,char *filename, ExceptionInfo *exception) { int file; const unsigned char *p; size_t i; size_t length, quantum; ssize_t count; struct stat file_stats; unsigned char *buffer; assert(image != (Image *) NULL); assert(image->signature == MagickCoreSignature); assert(image->blob != (BlobInfo *) NULL); assert(image->blob->type != UndefinedStream); if (image->debug != MagickFalse) (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",filename); assert(filename != (const char *) NULL); if (*filename == '\0') file=AcquireUniqueFileResource(filename); else if (LocaleCompare(filename,"-") == 0) file=fileno(stdout); else file=open_utf8(filename,O_RDWR | O_CREAT | O_EXCL | O_BINARY,S_MODE); if (file == -1) { ThrowFileException(exception,BlobError,"UnableToWriteBlob",filename); return(MagickFalse); } quantum=(size_t) MagickMaxBufferExtent; if ((fstat(file,&file_stats) == 0) && (file_stats.st_size > 0)) quantum=(size_t) MagickMin(file_stats.st_size,MagickMaxBufferExtent); buffer=(unsigned char *) AcquireQuantumMemory(quantum,sizeof(*buffer)); if (buffer == (unsigned char *) NULL) { file=close(file)-1; (void) ThrowMagickException(exception,GetMagickModule(), ResourceLimitError,"MemoryAllocationError","`%s'",filename); return(MagickFalse); } length=0; p=(const unsigned char *) ReadBlobStream(image,quantum,buffer,&count); for (i=0; count > 0; ) { length=(size_t) count; for (i=0; i < length; i+=count) { count=write(file,p+i,(size_t) (length-i)); if (count <= 0) { count=0; if (errno != EINTR) break; } } if (i < length) break; p=(const unsigned char *) ReadBlobStream(image,quantum,buffer,&count); } if (LocaleCompare(filename,"-") != 0) file=close(file); buffer=(unsigned char *) RelinquishMagickMemory(buffer); if ((file == -1) || (i < length)) { ThrowFileException(exception,BlobError,"UnableToWriteBlob",filename); return(MagickFalse); } return(MagickTrue); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % % I m a g e s T o B l o b % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % ImagesToBlob() implements direct to memory image formats. It returns the % image sequence as a blob and its length. The magick member of the ImageInfo % structure determines the format of the returned blob (GIF, JPEG, PNG, etc.) % % Note, some image formats do not permit multiple images to the same image % stream (e.g. JPEG). in this instance, just the first image of the % sequence is returned as a blob. % % The format of the ImagesToBlob method is: % % void *ImagesToBlob(const ImageInfo *image_info,Image *images, % size_t *length,ExceptionInfo *exception) % % A description of each parameter follows: % % o image_info: the image info. % % o images: the image list. % % o length: return the actual length of the blob. % % o exception: return any errors or warnings in this structure. % */ MagickExport void *ImagesToBlob(const ImageInfo *image_info,Image *images, size_t *length,ExceptionInfo *exception) { const MagickInfo *magick_info; ImageInfo *clone_info; MagickBooleanType status; void *blob; assert(image_info != (const ImageInfo *) NULL); assert(image_info->signature == MagickCoreSignature); if (image_info->debug != MagickFalse) (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s", image_info->filename); assert(images != (Image *) NULL); assert(images->signature == MagickCoreSignature); assert(exception != (ExceptionInfo *) NULL); *length=0; blob=(unsigned char *) NULL; clone_info=CloneImageInfo(image_info); (void) SetImageInfo(clone_info,(unsigned int) GetImageListLength(images), exception); if (*clone_info->magick != '\0') (void) CopyMagickString(images->magick,clone_info->magick,MagickPathExtent); magick_info=GetMagickInfo(images->magick,exception); if (magick_info == (const MagickInfo *) NULL) { (void) ThrowMagickException(exception,GetMagickModule(), MissingDelegateError,"NoDecodeDelegateForThisImageFormat","`%s'", images->magick); clone_info=DestroyImageInfo(clone_info); return(blob); } if (GetMagickAdjoin(magick_info) == MagickFalse) { clone_info=DestroyImageInfo(clone_info); return(ImageToBlob(image_info,images,length,exception)); } (void) CopyMagickString(clone_info->magick,images->magick,MagickPathExtent); if (GetMagickBlobSupport(magick_info) != MagickFalse) { /* Native blob support for this images format. */ clone_info->length=0; clone_info->blob=(void *) AcquireQuantumMemory(MagickMaxBlobExtent, sizeof(unsigned char)); if (clone_info->blob == (void *) NULL) (void) ThrowMagickException(exception,GetMagickModule(), ResourceLimitError,"MemoryAllocationFailed","`%s'",images->filename); else { (void) CloseBlob(images); images->blob->exempt=MagickTrue; *images->filename='\0'; status=WriteImages(clone_info,images,images->filename,exception); *length=images->blob->length; blob=DetachBlob(images->blob); if (blob == (void *) NULL) clone_info->blob=RelinquishMagickMemory(clone_info->blob); else if (status == MagickFalse) blob=RelinquishMagickMemory(blob); else blob=ResizeQuantumMemory(blob,*length+1,sizeof(unsigned char)); } } else { char filename[MagickPathExtent], unique[MagickPathExtent]; int file; /* Write file to disk in blob images format. */ file=AcquireUniqueFileResource(unique); if (file == -1) { ThrowFileException(exception,FileOpenError,"UnableToWriteBlob", image_info->filename); } else { clone_info->file=fdopen(file,"wb"); if (clone_info->file != (FILE *) NULL) { (void) FormatLocaleString(filename,MagickPathExtent,"%s:%s", images->magick,unique); status=WriteImages(clone_info,images,filename,exception); (void) CloseBlob(images); (void) fclose(clone_info->file); if (status != MagickFalse) blob=FileToBlob(unique,~0UL,length,exception); } (void) RelinquishUniqueFileResource(unique); } } clone_info=DestroyImageInfo(clone_info); return(blob); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + I m a g e s T o C u s t o m B l o b % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % ImagesToCustomStream() is the equivalent of WriteImages(), but writes the % formatted "file" to the custom stream rather than to an actual file. % % The format of the ImageToCustomStream method is: % % void ImagesToCustomStream(const ImageInfo *image_info,Image *images, % ExceptionInfo *exception) % % A description of each parameter follows: % % o image_info: the image info. % % o images: the image list. % % o exception: return any errors or warnings in this structure. % */ MagickExport void ImagesToCustomStream(const ImageInfo *image_info, Image *images,ExceptionInfo *exception) { const MagickInfo *magick_info; ImageInfo *clone_info; MagickBooleanType blob_support, status; assert(image_info != (const ImageInfo *) NULL); assert(image_info->signature == MagickCoreSignature); if (image_info->debug != MagickFalse) (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s", image_info->filename); assert(images != (Image *) NULL); assert(images->signature == MagickCoreSignature); assert(image_info->custom_stream != (CustomStreamInfo *) NULL); assert(image_info->custom_stream->signature == MagickCoreSignature); assert(image_info->custom_stream->reader != (CustomStreamHandler) NULL); assert(image_info->custom_stream->writer != (CustomStreamHandler) NULL); assert(exception != (ExceptionInfo *) NULL); clone_info=CloneImageInfo(image_info); (void) SetImageInfo(clone_info,(unsigned int) GetImageListLength(images), exception); if (*clone_info->magick != '\0') (void) CopyMagickString(images->magick,clone_info->magick,MagickPathExtent); magick_info=GetMagickInfo(images->magick,exception); if (magick_info == (const MagickInfo *) NULL) { (void) ThrowMagickException(exception,GetMagickModule(), MissingDelegateError,"NoEncodeDelegateForThisImageFormat","`%s'", images->magick); clone_info=DestroyImageInfo(clone_info); return; } (void) CopyMagickString(clone_info->magick,images->magick,MagickPathExtent); blob_support=GetMagickBlobSupport(magick_info); if ((blob_support != MagickFalse) && (GetMagickEncoderSeekableStream(magick_info) != MagickFalse)) { if ((clone_info->custom_stream->seeker == (CustomStreamSeeker) NULL) || (clone_info->custom_stream->teller == (CustomStreamTeller) NULL)) blob_support=MagickFalse; } if (blob_support != MagickFalse) { /* Native blob support for this image format. */ (void) CloseBlob(images); *images->filename='\0'; (void) WriteImages(clone_info,images,images->filename,exception); (void) CloseBlob(images); } else { char filename[MagickPathExtent], unique[MagickPathExtent]; int file; unsigned char *blob; /* Write file to disk in blob image format. */ clone_info->custom_stream=(CustomStreamInfo *) NULL; blob=(unsigned char *) AcquireQuantumMemory(MagickMaxBufferExtent, sizeof(*blob)); if (blob == (unsigned char *) NULL) { ThrowFileException(exception,BlobError,"UnableToWriteBlob", image_info->filename); clone_info=DestroyImageInfo(clone_info); return; } file=AcquireUniqueFileResource(unique); if (file == -1) { ThrowFileException(exception,BlobError,"UnableToWriteBlob", image_info->filename); blob=(unsigned char *) RelinquishMagickMemory(blob); clone_info=DestroyImageInfo(clone_info); return; } clone_info->file=fdopen(file,"wb+"); if (clone_info->file != (FILE *) NULL) { ssize_t count; (void) FormatLocaleString(filename,MagickPathExtent,"%s:%s", images->magick,unique); status=WriteImages(clone_info,images,filename,exception); (void) CloseBlob(images); if (status != MagickFalse) { (void) fseek(clone_info->file,0,SEEK_SET); count=(ssize_t) MagickMaxBufferExtent; while (count == (ssize_t) MagickMaxBufferExtent) { count=(ssize_t) fread(blob,sizeof(*blob),MagickMaxBufferExtent, clone_info->file); (void) image_info->custom_stream->writer(blob,(size_t) count, image_info->custom_stream->data); } } (void) fclose(clone_info->file); } blob=(unsigned char *) RelinquishMagickMemory(blob); (void) RelinquishUniqueFileResource(unique); } clone_info=DestroyImageInfo(clone_info); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % % I n j e c t I m a g e B l o b % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % InjectImageBlob() injects the image with a copy of itself in the specified % format (e.g. inject JPEG into a PDF image). % % The format of the InjectImageBlob method is: % % MagickBooleanType InjectImageBlob(const ImageInfo *image_info, % Image *image,Image *inject_image,const char *format, % ExceptionInfo *exception) % % A description of each parameter follows: % % o image_info: the image info.. % % o image: the image. % % o inject_image: inject into the image stream. % % o format: the image format. % % o exception: return any errors or warnings in this structure. % */ MagickExport MagickBooleanType InjectImageBlob(const ImageInfo *image_info, Image *image,Image *inject_image,const char *format,ExceptionInfo *exception) { char filename[MagickPathExtent]; FILE *unique_file; Image *byte_image; ImageInfo *write_info; int file; MagickBooleanType status; ssize_t i; size_t quantum; ssize_t count; struct stat file_stats; unsigned char *buffer; /* Write inject image to a temporary file. */ assert(image_info != (ImageInfo *) NULL); assert(image_info->signature == MagickCoreSignature); assert(image != (Image *) NULL); assert(image->signature == MagickCoreSignature); if (image->debug != MagickFalse) (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",image->filename); assert(inject_image != (Image *) NULL); assert(inject_image->signature == MagickCoreSignature); assert(exception != (ExceptionInfo *) NULL); unique_file=(FILE *) NULL; file=AcquireUniqueFileResource(filename); if (file != -1) unique_file=fdopen(file,"wb"); if ((file == -1) || (unique_file == (FILE *) NULL)) { (void) CopyMagickString(image->filename,filename,MagickPathExtent); ThrowFileException(exception,FileOpenError,"UnableToCreateTemporaryFile", image->filename); return(MagickFalse); } byte_image=CloneImage(inject_image,0,0,MagickFalse,exception); if (byte_image == (Image *) NULL) { (void) fclose(unique_file); (void) RelinquishUniqueFileResource(filename); return(MagickFalse); } (void) FormatLocaleString(byte_image->filename,MagickPathExtent,"%s:%s", format,filename); DestroyBlob(byte_image); byte_image->blob=CloneBlobInfo((BlobInfo *) NULL); write_info=CloneImageInfo(image_info); SetImageInfoFile(write_info,unique_file); status=WriteImage(write_info,byte_image,exception); write_info=DestroyImageInfo(write_info); byte_image=DestroyImage(byte_image); (void) fclose(unique_file); if (status == MagickFalse) { (void) RelinquishUniqueFileResource(filename); return(MagickFalse); } /* Inject into image stream. */ file=open_utf8(filename,O_RDONLY | O_BINARY,0); if (file == -1) { (void) RelinquishUniqueFileResource(filename); ThrowFileException(exception,FileOpenError,"UnableToOpenFile", image_info->filename); return(MagickFalse); } quantum=(size_t) MagickMaxBufferExtent; if ((fstat(file,&file_stats) == 0) && (file_stats.st_size > 0)) quantum=(size_t) MagickMin(file_stats.st_size,MagickMaxBufferExtent); buffer=(unsigned char *) AcquireQuantumMemory(quantum,sizeof(*buffer)); if (buffer == (unsigned char *) NULL) { (void) RelinquishUniqueFileResource(filename); file=close(file); ThrowBinaryException(ResourceLimitError,"MemoryAllocationFailed", image->filename); } for (i=0; ; i+=count) { count=read(file,buffer,quantum); if (count <= 0) { count=0; if (errno != EINTR) break; } status=WriteBlobStream(image,(size_t) count,buffer) == count ? MagickTrue : MagickFalse; } file=close(file); if (file == -1) ThrowFileException(exception,FileOpenError,"UnableToWriteBlob",filename); (void) RelinquishUniqueFileResource(filename); buffer=(unsigned char *) RelinquishMagickMemory(buffer); return(status); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % % I s B l o b E x e m p t % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % IsBlobExempt() returns true if the blob is exempt. % % The format of the IsBlobExempt method is: % % MagickBooleanType IsBlobExempt(const Image *image) % % A description of each parameter follows: % % o image: the image. % */ MagickExport MagickBooleanType IsBlobExempt(const Image *image) { assert(image != (const Image *) NULL); assert(image->signature == MagickCoreSignature); if (image->debug != MagickFalse) (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",image->filename); return(image->blob->exempt); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % % I s B l o b S e e k a b l e % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % IsBlobSeekable() returns true if the blob is seekable. % % The format of the IsBlobSeekable method is: % % MagickBooleanType IsBlobSeekable(const Image *image) % % A description of each parameter follows: % % o image: the image. % */ MagickExport MagickBooleanType IsBlobSeekable(const Image *image) { BlobInfo *magick_restrict blob_info; assert(image != (const Image *) NULL); assert(image->signature == MagickCoreSignature); if (image->debug != MagickFalse) (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",image->filename); blob_info=image->blob; switch (blob_info->type) { case BlobStream: return(MagickTrue); case FileStream: { int status; if (blob_info->file_info.file == (FILE *) NULL) return(MagickFalse); status=fseek(blob_info->file_info.file,0,SEEK_CUR); return(status == -1 ? MagickFalse : MagickTrue); } case ZipStream: { #if defined(MAGICKCORE_ZLIB_DELEGATE) MagickOffsetType offset; if (blob_info->file_info.gzfile == (gzFile) NULL) return(MagickFalse); offset=gzseek(blob_info->file_info.gzfile,0,SEEK_CUR); return(offset < 0 ? MagickFalse : MagickTrue); #else break; #endif } case UndefinedStream: case BZipStream: case FifoStream: case PipeStream: case StandardStream: break; case CustomStream: { if ((blob_info->custom_stream->seeker != (CustomStreamSeeker) NULL) && (blob_info->custom_stream->teller != (CustomStreamTeller) NULL)) return(MagickTrue); break; } default: break; } return(MagickFalse); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % % I s B l o b T e m p o r a r y % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % IsBlobTemporary() returns true if the blob is temporary. % % The format of the IsBlobTemporary method is: % % MagickBooleanType IsBlobTemporary(const Image *image) % % A description of each parameter follows: % % o image: the image. % */ MagickExport MagickBooleanType IsBlobTemporary(const Image *image) { assert(image != (const Image *) NULL); assert(image->signature == MagickCoreSignature); if (image->debug != MagickFalse) (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",image->filename); return(image->blob->temporary); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + M a p B l o b % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % MapBlob() creates a mapping from a file to a binary large object. % % The format of the MapBlob method is: % % void *MapBlob(int file,const MapMode mode,const MagickOffsetType offset, % const size_t length) % % A description of each parameter follows: % % o file: map this file descriptor. % % o mode: ReadMode, WriteMode, or IOMode. % % o offset: starting at this offset within the file. % % o length: the length of the mapping is returned in this pointer. % */ MagickExport void *MapBlob(int file,const MapMode mode, const MagickOffsetType offset,const size_t length) { #if defined(MAGICKCORE_HAVE_MMAP) int flags, protection; void *map; /* Map file. */ flags=0; if (file == -1) #if defined(MAP_ANONYMOUS) flags|=MAP_ANONYMOUS; #else return(NULL); #endif switch (mode) { case ReadMode: default: { protection=PROT_READ; flags|=MAP_PRIVATE; break; } case WriteMode: { protection=PROT_WRITE; flags|=MAP_SHARED; break; } case IOMode: { protection=PROT_READ | PROT_WRITE; flags|=MAP_SHARED; break; } } #if !defined(MAGICKCORE_HAVE_HUGEPAGES) || !defined(MAP_HUGETLB) map=mmap((char *) NULL,length,protection,flags,file,offset); #else map=mmap((char *) NULL,length,protection,flags | MAP_HUGETLB,file,offset); if (map == MAP_FAILED) map=mmap((char *) NULL,length,protection,flags,file,offset); #endif if (map == MAP_FAILED) return(NULL); return(map); #else (void) file; (void) mode; (void) offset; (void) length; return(NULL); #endif } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + M S B O r d e r L o n g % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % MSBOrderLong() converts a least-significant byte first buffer of integers to % most-significant byte first. % % The format of the MSBOrderLong method is: % % void MSBOrderLong(unsigned char *buffer,const size_t length) % % A description of each parameter follows. % % o buffer: Specifies a pointer to a buffer of integers. % % o length: Specifies the length of the buffer. % */ MagickExport void MSBOrderLong(unsigned char *buffer,const size_t length) { int c; unsigned char *p, *q; assert(buffer != (unsigned char *) NULL); q=buffer+length; while (buffer < q) { p=buffer+3; c=(int) (*p); *p=(*buffer); *buffer++=(unsigned char) c; p=buffer+1; c=(int) (*p); *p=(*buffer); *buffer++=(unsigned char) c; buffer+=2; } } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + M S B O r d e r S h o r t % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % MSBOrderShort() converts a least-significant byte first buffer of integers % to most-significant byte first. % % The format of the MSBOrderShort method is: % % void MSBOrderShort(unsigned char *p,const size_t length) % % A description of each parameter follows. % % o p: Specifies a pointer to a buffer of integers. % % o length: Specifies the length of the buffer. % */ MagickExport void MSBOrderShort(unsigned char *p,const size_t length) { int c; unsigned char *q; assert(p != (unsigned char *) NULL); q=p+length; while (p < q) { c=(int) (*p); *p=(*(p+1)); p++; *p++=(unsigned char) c; } } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + O p e n B l o b % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % OpenBlob() opens a file associated with the image. A file name of '-' sets % the file to stdin for type 'r' and stdout for type 'w'. If the filename % suffix is '.gz' or '.Z', the image is decompressed for type 'r' and % compressed for type 'w'. If the filename prefix is '|', it is piped to or % from a system command. % % The format of the OpenBlob method is: % % MagickBooleanType OpenBlob(const ImageInfo *image_info,Image *image, % const BlobMode mode,ExceptionInfo *exception) % % A description of each parameter follows: % % o image_info: the image info. % % o image: the image. % % o mode: the mode for opening the file. % */ static inline MagickBooleanType SetStreamBuffering(const ImageInfo *image_info, Image *image) { const char *option; int status; size_t size; size=MagickMinBufferExtent; option=GetImageOption(image_info,"stream:buffer-size"); if (option != (const char *) NULL) size=StringToUnsignedLong(option); status=setvbuf(image->blob->file_info.file,(char *) NULL,size == 0 ? _IONBF : _IOFBF,size); return(status == 0 ? MagickTrue : MagickFalse); } MagickExport MagickBooleanType OpenBlob(const ImageInfo *image_info, Image *image,const BlobMode mode,ExceptionInfo *exception) { BlobInfo *magick_restrict blob_info; char extension[MagickPathExtent], filename[MagickPathExtent]; const char *type; MagickBooleanType status; PolicyRights rights; assert(image_info != (ImageInfo *) NULL); assert(image_info->signature == MagickCoreSignature); if (image_info->debug != MagickFalse) (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s", image_info->filename); assert(image != (Image *) NULL); assert(image->signature == MagickCoreSignature); blob_info=image->blob; if (image_info->blob != (void *) NULL) { if (image_info->stream != (StreamHandler) NULL) blob_info->stream=(StreamHandler) image_info->stream; AttachBlob(blob_info,image_info->blob,image_info->length); return(MagickTrue); } if ((image_info->custom_stream != (CustomStreamInfo *) NULL) && (*image->filename == '\0')) { blob_info->type=CustomStream; blob_info->custom_stream=image_info->custom_stream; return(MagickTrue); } (void) DetachBlob(blob_info); blob_info->mode=mode; switch (mode) { default: type="r"; break; case ReadBlobMode: type="r"; break; case ReadBinaryBlobMode: type="rb"; break; case WriteBlobMode: type="w"; break; case WriteBinaryBlobMode: type="w+b"; break; case AppendBlobMode: type="a"; break; case AppendBinaryBlobMode: type="a+b"; break; } if (*type != 'r') blob_info->synchronize=image_info->synchronize; if (image_info->stream != (StreamHandler) NULL) { blob_info->stream=image_info->stream; if (*type == 'w') { blob_info->type=FifoStream; return(MagickTrue); } } /* Open image file. */ *filename='\0'; (void) CopyMagickString(filename,image->filename,MagickPathExtent); rights=ReadPolicyRights; if (*type == 'w') rights=WritePolicyRights; if (IsRightsAuthorized(PathPolicyDomain,rights,filename) == MagickFalse) { errno=EPERM; (void) ThrowMagickException(exception,GetMagickModule(),PolicyError, "NotAuthorized","`%s'",filename); return(MagickFalse); } if ((LocaleCompare(filename,"-") == 0) || ((*filename == '\0') && (image_info->file == (FILE *) NULL))) { blob_info->file_info.file=(*type == 'r') ? stdin : stdout; #if defined(MAGICKCORE_WINDOWS_SUPPORT) || defined(__OS2__) if (strchr(type,'b') != (char *) NULL) setmode(fileno(blob_info->file_info.file),_O_BINARY); #endif blob_info->type=StandardStream; blob_info->exempt=MagickTrue; return(SetStreamBuffering(image_info,image)); } if ((LocaleNCompare(filename,"fd:",3) == 0) && (IsGeometry(filename+3) != MagickFalse)) { char fileMode[MagickPathExtent]; *fileMode =(*type); fileMode[1]='\0'; blob_info->file_info.file=fdopen(StringToLong(filename+3),fileMode); if (blob_info->file_info.file == (FILE *) NULL) { ThrowFileException(exception,BlobError,"UnableToOpenBlob",filename); return(MagickFalse); } #if defined(MAGICKCORE_WINDOWS_SUPPORT) || defined(__OS2__) if (strchr(type,'b') != (char *) NULL) setmode(fileno(blob_info->file_info.file),_O_BINARY); #endif blob_info->type=FileStream; blob_info->exempt=MagickTrue; return(SetStreamBuffering(image_info,image)); } #if defined(MAGICKCORE_HAVE_POPEN) && defined(MAGICKCORE_PIPES_SUPPORT) if (*filename == '|') { char fileMode[MagickPathExtent], *sanitize_command; /* Pipe image to or from a system command. */ #if defined(SIGPIPE) if (*type == 'w') (void) signal(SIGPIPE,SIG_IGN); #endif *fileMode =(*type); fileMode[1]='\0'; sanitize_command=SanitizeString(filename+1); blob_info->file_info.file=(FILE *) popen_utf8(sanitize_command,fileMode); sanitize_command=DestroyString(sanitize_command); if (blob_info->file_info.file == (FILE *) NULL) { ThrowFileException(exception,BlobError,"UnableToOpenBlob",filename); return(MagickFalse); } blob_info->type=PipeStream; blob_info->exempt=MagickTrue; return(SetStreamBuffering(image_info,image)); } #endif status=GetPathAttributes(filename,&blob_info->properties); #if defined(S_ISFIFO) if ((status != MagickFalse) && S_ISFIFO(blob_info->properties.st_mode)) { blob_info->file_info.file=(FILE *) fopen_utf8(filename,type); if (blob_info->file_info.file == (FILE *) NULL) { ThrowFileException(exception,BlobError,"UnableToOpenBlob",filename); return(MagickFalse); } blob_info->type=FileStream; blob_info->exempt=MagickTrue; return(SetStreamBuffering(image_info,image)); } #endif GetPathComponent(image->filename,ExtensionPath,extension); if (*type == 'w') { (void) CopyMagickString(filename,image->filename,MagickPathExtent); if ((image_info->adjoin == MagickFalse) || (strchr(filename,'%') != (char *) NULL)) { /* Form filename for multi-part images. */ (void) InterpretImageFilename(image_info,image,image->filename,(int) image->scene,filename,exception); if ((LocaleCompare(filename,image->filename) == 0) && ((GetPreviousImageInList(image) != (Image *) NULL) || (GetNextImageInList(image) != (Image *) NULL))) { char path[MagickPathExtent]; GetPathComponent(image->filename,RootPath,path); if (*extension == '\0') (void) FormatLocaleString(filename,MagickPathExtent,"%s-%.20g", path,(double) image->scene); else (void) FormatLocaleString(filename,MagickPathExtent, "%s-%.20g.%s",path,(double) image->scene,extension); } (void) CopyMagickString(image->filename,filename,MagickPathExtent); #if defined(macintosh) SetApplicationType(filename,image_info->magick,'8BIM'); #endif } } if (image_info->file != (FILE *) NULL) { blob_info->file_info.file=image_info->file; blob_info->type=FileStream; blob_info->exempt=MagickTrue; } else if (*type == 'r') { blob_info->file_info.file=(FILE *) fopen_utf8(filename,type); if (blob_info->file_info.file != (FILE *) NULL) { size_t count; unsigned char magick[3]; blob_info->type=FileStream; (void) SetStreamBuffering(image_info,image); (void) memset(magick,0,sizeof(magick)); count=fread(magick,1,sizeof(magick),blob_info->file_info.file); (void) fseek(blob_info->file_info.file,-((off_t) count),SEEK_CUR); #if defined(MAGICKCORE_POSIX_SUPPORT) (void) fflush(blob_info->file_info.file); #endif (void) LogMagickEvent(BlobEvent,GetMagickModule(), " read %.20g magic header bytes",(double) count); #if defined(MAGICKCORE_ZLIB_DELEGATE) if (((int) magick[0] == 0x1F) && ((int) magick[1] == 0x8B) && ((int) magick[2] == 0x08)) { if (blob_info->file_info.file != (FILE *) NULL) (void) fclose(blob_info->file_info.file); blob_info->file_info.file=(FILE *) NULL; blob_info->file_info.gzfile=gzopen(filename,"rb"); if (blob_info->file_info.gzfile != (gzFile) NULL) blob_info->type=ZipStream; } #endif #if defined(MAGICKCORE_BZLIB_DELEGATE) if (strncmp((char *) magick,"BZh",3) == 0) { if (blob_info->file_info.file != (FILE *) NULL) (void) fclose(blob_info->file_info.file); blob_info->file_info.file=(FILE *) NULL; blob_info->file_info.bzfile=BZ2_bzopen(filename,"r"); if (blob_info->file_info.bzfile != (BZFILE *) NULL) blob_info->type=BZipStream; } #endif if (blob_info->type == FileStream) { const MagickInfo *magick_info; ExceptionInfo *sans_exception; size_t length; sans_exception=AcquireExceptionInfo(); magick_info=GetMagickInfo(image_info->magick,sans_exception); sans_exception=DestroyExceptionInfo(sans_exception); length=(size_t) blob_info->properties.st_size; if ((magick_info != (const MagickInfo *) NULL) && (GetMagickBlobSupport(magick_info) != MagickFalse) && (length > MagickMaxBufferExtent) && (AcquireMagickResource(MapResource,length) != MagickFalse)) { void *blob; blob=MapBlob(fileno(blob_info->file_info.file),ReadMode,0, length); if (blob == (void *) NULL) RelinquishMagickResource(MapResource,length); else { /* Format supports blobs-- use memory-mapped I/O. */ if (image_info->file != (FILE *) NULL) blob_info->exempt=MagickFalse; else { (void) fclose(blob_info->file_info.file); blob_info->file_info.file=(FILE *) NULL; } AttachBlob(blob_info,blob,length); blob_info->mapped=MagickTrue; } } } } } else #if defined(MAGICKCORE_ZLIB_DELEGATE) if ((LocaleCompare(extension,"Z") == 0) || (LocaleCompare(extension,"gz") == 0) || (LocaleCompare(extension,"wmz") == 0) || (LocaleCompare(extension,"svgz") == 0)) { blob_info->file_info.gzfile=gzopen(filename,"wb"); if (blob_info->file_info.gzfile != (gzFile) NULL) blob_info->type=ZipStream; } else #endif #if defined(MAGICKCORE_BZLIB_DELEGATE) if (LocaleCompare(extension,"bz2") == 0) { blob_info->file_info.bzfile=BZ2_bzopen(filename,"w"); if (blob_info->file_info.bzfile != (BZFILE *) NULL) blob_info->type=BZipStream; } else #endif { blob_info->file_info.file=(FILE *) fopen_utf8(filename,type); if (blob_info->file_info.file != (FILE *) NULL) { blob_info->type=FileStream; (void) SetStreamBuffering(image_info,image); } } blob_info->status=MagickFalse; blob_info->error_number=MagickFalse; if (blob_info->type != UndefinedStream) blob_info->size=GetBlobSize(image); else { ThrowFileException(exception,BlobError,"UnableToOpenBlob",filename); return(MagickFalse); } return(MagickTrue); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + P i n g B l o b % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % PingBlob() returns all the attributes of an image or image sequence except % for the pixels. It is much faster and consumes far less memory than % BlobToImage(). On failure, a NULL image is returned and exception % describes the reason for the failure. % % The format of the PingBlob method is: % % Image *PingBlob(const ImageInfo *image_info,const void *blob, % const size_t length,ExceptionInfo *exception) % % A description of each parameter follows: % % o image_info: the image info. % % o blob: the address of a character stream in one of the image formats % understood by ImageMagick. % % o length: This size_t integer reflects the length in bytes of the blob. % % o exception: return any errors or warnings in this structure. % */ #if defined(__cplusplus) || defined(c_plusplus) extern "C" { #endif static size_t PingStream(const Image *magick_unused(image), const void *magick_unused(pixels),const size_t columns) { magick_unreferenced(image); magick_unreferenced(pixels); return(columns); } #if defined(__cplusplus) || defined(c_plusplus) } #endif MagickExport Image *PingBlob(const ImageInfo *image_info,const void *blob, const size_t length,ExceptionInfo *exception) { const MagickInfo *magick_info; Image *image; ImageInfo *clone_info, *ping_info; MagickBooleanType status; assert(image_info != (ImageInfo *) NULL); assert(image_info->signature == MagickCoreSignature); if (image_info->debug != MagickFalse) (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s", image_info->filename); assert(exception != (ExceptionInfo *) NULL); if ((blob == (const void *) NULL) || (length == 0)) { (void) ThrowMagickException(exception,GetMagickModule(),BlobError, "ZeroLengthBlobNotPermitted","`%s'",image_info->filename); return((Image *) NULL); } ping_info=CloneImageInfo(image_info); ping_info->blob=(void *) blob; ping_info->length=length; ping_info->ping=MagickTrue; if (*ping_info->magick == '\0') (void) SetImageInfo(ping_info,0,exception); magick_info=GetMagickInfo(ping_info->magick,exception); if (magick_info == (const MagickInfo *) NULL) { (void) ThrowMagickException(exception,GetMagickModule(), MissingDelegateError,"NoDecodeDelegateForThisImageFormat","`%s'", ping_info->magick); ping_info=DestroyImageInfo(ping_info); return((Image *) NULL); } if (GetMagickBlobSupport(magick_info) != MagickFalse) { char filename[MagickPathExtent]; /* Native blob support for this image format. */ (void) CopyMagickString(filename,ping_info->filename,MagickPathExtent); (void) FormatLocaleString(ping_info->filename,MagickPathExtent,"%s:%s", ping_info->magick,filename); image=ReadStream(ping_info,&PingStream,exception); if (image != (Image *) NULL) (void) DetachBlob(image->blob); ping_info=DestroyImageInfo(ping_info); return(image); } /* Write blob to a temporary file on disk. */ ping_info->blob=(void *) NULL; ping_info->length=0; *ping_info->filename='\0'; status=BlobToFile(ping_info->filename,blob,length,exception); if (status == MagickFalse) { (void) RelinquishUniqueFileResource(ping_info->filename); ping_info=DestroyImageInfo(ping_info); return((Image *) NULL); } clone_info=CloneImageInfo(ping_info); (void) FormatLocaleString(clone_info->filename,MagickPathExtent,"%s:%s", ping_info->magick,ping_info->filename); image=ReadStream(clone_info,&PingStream,exception); if (image != (Image *) NULL) { Image *images; /* Restore original filenames and image format. */ for (images=GetFirstImageInList(image); images != (Image *) NULL; ) { (void) CopyMagickString(images->filename,image_info->filename, MagickPathExtent); (void) CopyMagickString(images->magick_filename,image_info->filename, MagickPathExtent); (void) CopyMagickString(images->magick,magick_info->name, MagickPathExtent); images=GetNextImageInList(images); } } clone_info=DestroyImageInfo(clone_info); (void) RelinquishUniqueFileResource(ping_info->filename); ping_info=DestroyImageInfo(ping_info); return(image); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + R e a d B l o b % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % ReadBlob() reads data from the blob or image file and returns it. It % returns the number of bytes read. If length is zero, ReadBlob() returns % zero and has no other results. If length is greater than MAGICK_SSIZE_MAX, the % result is unspecified. % % The format of the ReadBlob method is: % % ssize_t ReadBlob(Image *image,const size_t length,void *data) % % A description of each parameter follows: % % o image: the image. % % o length: Specifies an integer representing the number of bytes to read % from the file. % % o data: Specifies an area to place the information requested from the % file. % */ MagickExport ssize_t ReadBlob(Image *image,const size_t length,void *data) { BlobInfo *magick_restrict blob_info; int c; unsigned char *q; ssize_t count; assert(image != (Image *) NULL); assert(image->signature == MagickCoreSignature); assert(image->blob != (BlobInfo *) NULL); assert(image->blob->type != UndefinedStream); if (length == 0) return(0); assert(data != (void *) NULL); blob_info=image->blob; count=0; q=(unsigned char *) data; switch (blob_info->type) { case UndefinedStream: break; case StandardStream: case FileStream: case PipeStream: { switch (length) { default: { count=(ssize_t) fread(q,1,length,blob_info->file_info.file); break; } case 4: { c=getc(blob_info->file_info.file); if (c == EOF) break; *q++=(unsigned char) c; count++; } case 3: { c=getc(blob_info->file_info.file); if (c == EOF) break; *q++=(unsigned char) c; count++; } case 2: { c=getc(blob_info->file_info.file); if (c == EOF) break; *q++=(unsigned char) c; count++; } case 1: { c=getc(blob_info->file_info.file); if (c == EOF) break; *q++=(unsigned char) c; count++; } case 0: break; } if ((count != (ssize_t) length) && (ferror(blob_info->file_info.file) != 0)) ThrowBlobException(blob_info); break; } case ZipStream: { #if defined(MAGICKCORE_ZLIB_DELEGATE) int status; switch (length) { default: { ssize_t i; for (i=0; i < (ssize_t) length; i+=count) { count=(ssize_t) gzread(blob_info->file_info.gzfile,q+i, (unsigned int) MagickMin(length-i,MagickMaxBufferExtent)); if (count <= 0) { count=0; if (errno != EINTR) break; } } count=i; break; } case 4: { c=gzgetc(blob_info->file_info.gzfile); if (c == EOF) break; *q++=(unsigned char) c; count++; } case 3: { c=gzgetc(blob_info->file_info.gzfile); if (c == EOF) break; *q++=(unsigned char) c; count++; } case 2: { c=gzgetc(blob_info->file_info.gzfile); if (c == EOF) break; *q++=(unsigned char) c; count++; } case 1: { c=gzgetc(blob_info->file_info.gzfile); if (c == EOF) break; *q++=(unsigned char) c; count++; } case 0: break; } status=Z_OK; (void) gzerror(blob_info->file_info.gzfile,&status); if ((count != (ssize_t) length) && (status != Z_OK)) ThrowBlobException(blob_info); if (blob_info->eof == MagickFalse) blob_info->eof=gzeof(blob_info->file_info.gzfile) != 0 ? MagickTrue : MagickFalse; #endif break; } case BZipStream: { #if defined(MAGICKCORE_BZLIB_DELEGATE) int status; ssize_t i; for (i=0; i < (ssize_t) length; i+=count) { count=(ssize_t) BZ2_bzread(blob_info->file_info.bzfile,q+i, (unsigned int) MagickMin(length-i,MagickMaxBufferExtent)); if (count <= 0) { count=0; if (errno != EINTR) break; } } count=i; status=BZ_OK; (void) BZ2_bzerror(blob_info->file_info.bzfile,&status); if ((count != (ssize_t) length) && (status != BZ_OK)) ThrowBlobException(blob_info); #endif break; } case FifoStream: break; case BlobStream: { const unsigned char *p; if (blob_info->offset >= (MagickOffsetType) blob_info->length) { blob_info->eof=MagickTrue; break; } p=blob_info->data+blob_info->offset; count=(ssize_t) MagickMin((MagickOffsetType) length,(MagickOffsetType) blob_info->length-blob_info->offset); blob_info->offset+=count; if (count != (ssize_t) length) blob_info->eof=MagickTrue; (void) memcpy(q,p,(size_t) count); break; } case CustomStream: { if (blob_info->custom_stream->reader != (CustomStreamHandler) NULL) count=blob_info->custom_stream->reader(q,length, blob_info->custom_stream->data); break; } } return(count); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + R e a d B l o b B y t e % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % ReadBlobByte() reads a single byte from the image file and returns it. % % The format of the ReadBlobByte method is: % % int ReadBlobByte(Image *image) % % A description of each parameter follows. % % o image: the image. % */ MagickExport int ReadBlobByte(Image *image) { BlobInfo *magick_restrict blob_info; const unsigned char *p; unsigned char buffer[1]; assert(image != (Image *) NULL); assert(image->signature == MagickCoreSignature); assert(image->blob != (BlobInfo *) NULL); assert(image->blob->type != UndefinedStream); blob_info=image->blob; switch (blob_info->type) { case StandardStream: case FileStream: case PipeStream: { int c; p=(const unsigned char *) buffer; c=getc(blob_info->file_info.file); if (c == EOF) return(EOF); *buffer=(unsigned char) c; break; } default: { ssize_t count; p=(const unsigned char *) ReadBlobStream(image,1,buffer,&count); if (count != 1) return(EOF); break; } } return((int) (*p)); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + R e a d B l o b D o u b l e % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % ReadBlobDouble() reads a double value as a 64-bit quantity in the byte-order % specified by the endian member of the image structure. % % The format of the ReadBlobDouble method is: % % double ReadBlobDouble(Image *image) % % A description of each parameter follows. % % o image: the image. % */ MagickExport double ReadBlobDouble(Image *image) { union { MagickSizeType unsigned_value; double double_value; } quantum; quantum.double_value=0.0; quantum.unsigned_value=ReadBlobLongLong(image); return(quantum.double_value); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + R e a d B l o b F l o a t % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % ReadBlobFloat() reads a float value as a 32-bit quantity in the byte-order % specified by the endian member of the image structure. % % The format of the ReadBlobFloat method is: % % float ReadBlobFloat(Image *image) % % A description of each parameter follows. % % o image: the image. % */ MagickExport float ReadBlobFloat(Image *image) { union { unsigned int unsigned_value; float float_value; } quantum; quantum.float_value=0.0; quantum.unsigned_value=ReadBlobLong(image); return(quantum.float_value); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + R e a d B l o b L o n g % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % ReadBlobLong() reads a unsigned int value as a 32-bit quantity in the % byte-order specified by the endian member of the image structure. % % The format of the ReadBlobLong method is: % % unsigned int ReadBlobLong(Image *image) % % A description of each parameter follows. % % o image: the image. % */ MagickExport unsigned int ReadBlobLong(Image *image) { const unsigned char *p; ssize_t count; unsigned char buffer[4]; unsigned int value; assert(image != (Image *) NULL); assert(image->signature == MagickCoreSignature); *buffer='\0'; p=(const unsigned char *) ReadBlobStream(image,4,buffer,&count); if (count != 4) return(0UL); if (image->endian == LSBEndian) { value=(unsigned int) (*p++); value|=(unsigned int) (*p++) << 8; value|=(unsigned int) (*p++) << 16; value|=(unsigned int) (*p++) << 24; return(value); } value=(unsigned int) (*p++) << 24; value|=(unsigned int) (*p++) << 16; value|=(unsigned int) (*p++) << 8; value|=(unsigned int) (*p++); return(value); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + R e a d B l o b L o n g L o n g % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % ReadBlobLongLong() reads a long long value as a 64-bit quantity in the % byte-order specified by the endian member of the image structure. % % The format of the ReadBlobLongLong method is: % % MagickSizeType ReadBlobLongLong(Image *image) % % A description of each parameter follows. % % o image: the image. % */ MagickExport MagickSizeType ReadBlobLongLong(Image *image) { MagickSizeType value; const unsigned char *p; ssize_t count; unsigned char buffer[8]; assert(image != (Image *) NULL); assert(image->signature == MagickCoreSignature); *buffer='\0'; p=(const unsigned char *) ReadBlobStream(image,8,buffer,&count); if (count != 8) return(MagickULLConstant(0)); if (image->endian == LSBEndian) { value=(MagickSizeType) (*p++); value|=(MagickSizeType) (*p++) << 8; value|=(MagickSizeType) (*p++) << 16; value|=(MagickSizeType) (*p++) << 24; value|=(MagickSizeType) (*p++) << 32; value|=(MagickSizeType) (*p++) << 40; value|=(MagickSizeType) (*p++) << 48; value|=(MagickSizeType) (*p++) << 56; return(value); } value=(MagickSizeType) (*p++) << 56; value|=(MagickSizeType) (*p++) << 48; value|=(MagickSizeType) (*p++) << 40; value|=(MagickSizeType) (*p++) << 32; value|=(MagickSizeType) (*p++) << 24; value|=(MagickSizeType) (*p++) << 16; value|=(MagickSizeType) (*p++) << 8; value|=(MagickSizeType) (*p++); return(value); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + R e a d B l o b S h o r t % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % ReadBlobShort() reads a short value as a 16-bit quantity in the byte-order % specified by the endian member of the image structure. % % The format of the ReadBlobShort method is: % % unsigned short ReadBlobShort(Image *image) % % A description of each parameter follows. % % o image: the image. % */ MagickExport unsigned short ReadBlobShort(Image *image) { const unsigned char *p; unsigned short value; ssize_t count; unsigned char buffer[2]; assert(image != (Image *) NULL); assert(image->signature == MagickCoreSignature); *buffer='\0'; p=(const unsigned char *) ReadBlobStream(image,2,buffer,&count); if (count != 2) return((unsigned short) 0U); if (image->endian == LSBEndian) { value=(unsigned short) (*p++); value|=(unsigned short) (*p++) << 8; return(value); } value=(unsigned short) ((unsigned short) (*p++) << 8); value|=(unsigned short) (*p++); return(value); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + R e a d B l o b L S B L o n g % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % ReadBlobLSBLong() reads a unsigned int value as a 32-bit quantity in % least-significant byte first order. % % The format of the ReadBlobLSBLong method is: % % unsigned int ReadBlobLSBLong(Image *image) % % A description of each parameter follows. % % o image: the image. % */ MagickExport unsigned int ReadBlobLSBLong(Image *image) { const unsigned char *p; unsigned int value; ssize_t count; unsigned char buffer[4]; assert(image != (Image *) NULL); assert(image->signature == MagickCoreSignature); *buffer='\0'; p=(const unsigned char *) ReadBlobStream(image,4,buffer,&count); if (count != 4) return(0U); value=(unsigned int) (*p++); value|=(unsigned int) (*p++) << 8; value|=(unsigned int) (*p++) << 16; value|=(unsigned int) (*p++) << 24; return(value); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + R e a d B l o b L S B S i g n e d L o n g % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % ReadBlobLSBSignedLong() reads a signed int value as a 32-bit quantity in % least-significant byte first order. % % The format of the ReadBlobLSBSignedLong method is: % % signed int ReadBlobLSBSignedLong(Image *image) % % A description of each parameter follows. % % o image: the image. % */ MagickExport signed int ReadBlobLSBSignedLong(Image *image) { union { unsigned int unsigned_value; signed int signed_value; } quantum; quantum.unsigned_value=ReadBlobLSBLong(image); return(quantum.signed_value); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + R e a d B l o b L S B S h o r t % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % ReadBlobLSBShort() reads a short value as a 16-bit quantity in % least-significant byte first order. % % The format of the ReadBlobLSBShort method is: % % unsigned short ReadBlobLSBShort(Image *image) % % A description of each parameter follows. % % o image: the image. % */ MagickExport unsigned short ReadBlobLSBShort(Image *image) { const unsigned char *p; unsigned short value; ssize_t count; unsigned char buffer[2]; assert(image != (Image *) NULL); assert(image->signature == MagickCoreSignature); *buffer='\0'; p=(const unsigned char *) ReadBlobStream(image,2,buffer,&count); if (count != 2) return((unsigned short) 0U); value=(unsigned short) (*p++); value|=(unsigned short) (*p++) << 8; return(value); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + R e a d B l o b L S B S i g n e d S h o r t % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % ReadBlobLSBSignedShort() reads a signed short value as a 16-bit quantity in % least-significant byte-order. % % The format of the ReadBlobLSBSignedShort method is: % % signed short ReadBlobLSBSignedShort(Image *image) % % A description of each parameter follows. % % o image: the image. % */ MagickExport signed short ReadBlobLSBSignedShort(Image *image) { union { unsigned short unsigned_value; signed short signed_value; } quantum; quantum.unsigned_value=ReadBlobLSBShort(image); return(quantum.signed_value); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + R e a d B l o b M S B L o n g % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % ReadBlobMSBLong() reads a unsigned int value as a 32-bit quantity in % most-significant byte first order. % % The format of the ReadBlobMSBLong method is: % % unsigned int ReadBlobMSBLong(Image *image) % % A description of each parameter follows. % % o image: the image. % */ MagickExport unsigned int ReadBlobMSBLong(Image *image) { const unsigned char *p; unsigned int value; ssize_t count; unsigned char buffer[4]; assert(image != (Image *) NULL); assert(image->signature == MagickCoreSignature); *buffer='\0'; p=(const unsigned char *) ReadBlobStream(image,4,buffer,&count); if (count != 4) return(0UL); value=(unsigned int) (*p++) << 24; value|=(unsigned int) (*p++) << 16; value|=(unsigned int) (*p++) << 8; value|=(unsigned int) (*p++); return(value); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + R e a d B l o b M S B L o n g L o n g % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % ReadBlobMSBLongLong() reads a unsigned long long value as a 64-bit quantity % in most-significant byte first order. % % The format of the ReadBlobMSBLongLong method is: % % unsigned int ReadBlobMSBLongLong(Image *image) % % A description of each parameter follows. % % o image: the image. % */ MagickExport MagickSizeType ReadBlobMSBLongLong(Image *image) { const unsigned char *p; MagickSizeType value; ssize_t count; unsigned char buffer[8]; assert(image != (Image *) NULL); assert(image->signature == MagickCoreSignature); *buffer='\0'; p=(const unsigned char *) ReadBlobStream(image,8,buffer,&count); if (count != 8) return(MagickULLConstant(0)); value=(MagickSizeType) (*p++) << 56; value|=(MagickSizeType) (*p++) << 48; value|=(MagickSizeType) (*p++) << 40; value|=(MagickSizeType) (*p++) << 32; value|=(MagickSizeType) (*p++) << 24; value|=(MagickSizeType) (*p++) << 16; value|=(MagickSizeType) (*p++) << 8; value|=(MagickSizeType) (*p++); return(value); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + R e a d B l o b M S B S h o r t % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % ReadBlobMSBShort() reads a short value as a 16-bit quantity in % most-significant byte first order. % % The format of the ReadBlobMSBShort method is: % % unsigned short ReadBlobMSBShort(Image *image) % % A description of each parameter follows. % % o image: the image. % */ MagickExport unsigned short ReadBlobMSBShort(Image *image) { const unsigned char *p; unsigned short value; ssize_t count; unsigned char buffer[2]; assert(image != (Image *) NULL); assert(image->signature == MagickCoreSignature); *buffer='\0'; p=(const unsigned char *) ReadBlobStream(image,2,buffer,&count); if (count != 2) return((unsigned short) 0U); value=(unsigned short) ((*p++) << 8); value|=(unsigned short) (*p++); return((unsigned short) (value & 0xffff)); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + R e a d B l o b M S B S i g n e d L o n g % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % ReadBlobMSBSignedLong() reads a signed int value as a 32-bit quantity in % most-significant byte-order. % % The format of the ReadBlobMSBSignedLong method is: % % signed int ReadBlobMSBSignedLong(Image *image) % % A description of each parameter follows. % % o image: the image. % */ MagickExport signed int ReadBlobMSBSignedLong(Image *image) { union { unsigned int unsigned_value; signed int signed_value; } quantum; quantum.unsigned_value=ReadBlobMSBLong(image); return(quantum.signed_value); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + R e a d B l o b M S B S i g n e d S h o r t % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % ReadBlobMSBSignedShort() reads a signed short value as a 16-bit quantity in % most-significant byte-order. % % The format of the ReadBlobMSBSignedShort method is: % % signed short ReadBlobMSBSignedShort(Image *image) % % A description of each parameter follows. % % o image: the image. % */ MagickExport signed short ReadBlobMSBSignedShort(Image *image) { union { unsigned short unsigned_value; signed short signed_value; } quantum; quantum.unsigned_value=ReadBlobMSBShort(image); return(quantum.signed_value); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + R e a d B l o b S i g n e d L o n g % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % ReadBlobSignedLong() reads a signed int value as a 32-bit quantity in the % byte-order specified by the endian member of the image structure. % % The format of the ReadBlobSignedLong method is: % % signed int ReadBlobSignedLong(Image *image) % % A description of each parameter follows. % % o image: the image. % */ MagickExport signed int ReadBlobSignedLong(Image *image) { union { unsigned int unsigned_value; signed int signed_value; } quantum; quantum.unsigned_value=ReadBlobLong(image); return(quantum.signed_value); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + R e a d B l o b S i g n e d S h o r t % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % ReadBlobSignedShort() reads a signed short value as a 16-bit quantity in the % byte-order specified by the endian member of the image structure. % % The format of the ReadBlobSignedShort method is: % % signed short ReadBlobSignedShort(Image *image) % % A description of each parameter follows. % % o image: the image. % */ MagickExport signed short ReadBlobSignedShort(Image *image) { union { unsigned short unsigned_value; signed short signed_value; } quantum; quantum.unsigned_value=ReadBlobShort(image); return(quantum.signed_value); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + R e a d B l o b S t r e a m % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % ReadBlobStream() reads data from the blob or image file and returns it. It % returns a pointer to the data buffer you supply or to the image memory % buffer if its supported (zero-copy). If length is zero, ReadBlobStream() % returns a count of zero and has no other results. If length is greater than % MAGICK_SSIZE_MAX, the result is unspecified. % % The format of the ReadBlobStream method is: % % const void *ReadBlobStream(Image *image,const size_t length, % void *magick_restrict data,ssize_t *count) % % A description of each parameter follows: % % o image: the image. % % o length: Specifies an integer representing the number of bytes to read % from the file. % % o count: returns the number of bytes read. % % o data: Specifies an area to place the information requested from the % file. % */ MagickExport magick_hot_spot const void *ReadBlobStream(Image *image, const size_t length,void *magick_restrict data,ssize_t *count) { BlobInfo *magick_restrict blob_info; assert(image != (Image *) NULL); assert(image->signature == MagickCoreSignature); assert(image->blob != (BlobInfo *) NULL); assert(image->blob->type != UndefinedStream); assert(count != (ssize_t *) NULL); blob_info=image->blob; if (blob_info->type != BlobStream) { assert(data != NULL); *count=ReadBlob(image,length,(unsigned char *) data); return(data); } if (blob_info->offset >= (MagickOffsetType) blob_info->length) { *count=0; blob_info->eof=MagickTrue; return(data); } data=blob_info->data+blob_info->offset; *count=(ssize_t) MagickMin((MagickOffsetType) length,(MagickOffsetType) blob_info->length-blob_info->offset); blob_info->offset+=(*count); if (*count != (ssize_t) length) blob_info->eof=MagickTrue; return(data); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + R e a d B l o b S t r i n g % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % ReadBlobString() reads characters from a blob or file until a newline % character is read or an end-of-file condition is encountered. % % The format of the ReadBlobString method is: % % char *ReadBlobString(Image *image,char *string) % % A description of each parameter follows: % % o image: the image. % % o string: the address of a character buffer. % */ MagickExport char *ReadBlobString(Image *image,char *string) { int c; ssize_t i; assert(image != (Image *) NULL); assert(image->signature == MagickCoreSignature); for (i=0; i < (MagickPathExtent-1L); i++) { c=ReadBlobByte(image); if (c == EOF) { if (i == 0) return((char *) NULL); break; } string[i]=c; if (c == '\n') { if ((i > 0) && (string[i-1] == '\r')) i--; break; } } string[i]='\0'; return(string); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + R e f e r e n c e B l o b % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % ReferenceBlob() increments the reference count associated with the pixel % blob returning a pointer to the blob. % % The format of the ReferenceBlob method is: % % BlobInfo ReferenceBlob(BlobInfo *blob_info) % % A description of each parameter follows: % % o blob_info: the blob_info. % */ MagickExport BlobInfo *ReferenceBlob(BlobInfo *blob) { assert(blob != (BlobInfo *) NULL); assert(blob->signature == MagickCoreSignature); if (blob->debug != MagickFalse) (void) LogMagickEvent(TraceEvent,GetMagickModule(),"..."); LockSemaphoreInfo(blob->semaphore); blob->reference_count++; UnlockSemaphoreInfo(blob->semaphore); return(blob); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + S e e k B l o b % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % SeekBlob() sets the offset in bytes from the beginning of a blob or file % and returns the resulting offset. % % The format of the SeekBlob method is: % % MagickOffsetType SeekBlob(Image *image,const MagickOffsetType offset, % const int whence) % % A description of each parameter follows: % % o image: the image. % % o offset: Specifies an integer representing the offset in bytes. % % o whence: Specifies an integer representing how the offset is % treated relative to the beginning of the blob as follows: % % SEEK_SET Set position equal to offset bytes. % SEEK_CUR Set position to current location plus offset. % SEEK_END Set position to EOF plus offset. % */ MagickExport MagickOffsetType SeekBlob(Image *image, const MagickOffsetType offset,const int whence) { BlobInfo *magick_restrict blob_info; assert(image != (Image *) NULL); assert(image->signature == MagickCoreSignature); if (image->debug != MagickFalse) (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",image->filename); assert(image->blob != (BlobInfo *) NULL); assert(image->blob->type != UndefinedStream); blob_info=image->blob; switch (blob_info->type) { case UndefinedStream: break; case StandardStream: case PipeStream: return(-1); case FileStream: { if ((offset < 0) && (whence == SEEK_SET)) return(-1); if (fseek(blob_info->file_info.file,offset,whence) < 0) return(-1); blob_info->offset=TellBlob(image); break; } case ZipStream: { #if defined(MAGICKCORE_ZLIB_DELEGATE) if (gzseek(blob_info->file_info.gzfile,offset,whence) < 0) return(-1); #endif blob_info->offset=TellBlob(image); break; } case BZipStream: return(-1); case FifoStream: return(-1); case BlobStream: { switch (whence) { case SEEK_SET: default: { if (offset < 0) return(-1); blob_info->offset=offset; break; } case SEEK_CUR: { if (((offset > 0) && (blob_info->offset > (MAGICK_SSIZE_MAX-offset))) || ((offset < 0) && (blob_info->offset < (MAGICK_SSIZE_MIN-offset)))) { errno=EOVERFLOW; return(-1); } if ((blob_info->offset+offset) < 0) return(-1); blob_info->offset+=offset; break; } case SEEK_END: { if (((MagickOffsetType) blob_info->length+offset) < 0) return(-1); blob_info->offset=blob_info->length+offset; break; } } if (blob_info->offset < (MagickOffsetType) ((off_t) blob_info->length)) { blob_info->eof=MagickFalse; break; } if (blob_info->offset >= (MagickOffsetType) ((off_t) blob_info->extent)) return(-1); break; } case CustomStream: { if (blob_info->custom_stream->seeker == (CustomStreamSeeker) NULL) return(-1); blob_info->offset=blob_info->custom_stream->seeker(offset,whence, blob_info->custom_stream->data); break; } } return(blob_info->offset); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + S e t B l o b E x e m p t % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % SetBlobExempt() sets the blob exempt status. % % The format of the SetBlobExempt method is: % % MagickBooleanType SetBlobExempt(const Image *image, % const MagickBooleanType exempt) % % A description of each parameter follows: % % o image: the image. % % o exempt: Set to true if this blob is exempt from being closed. % */ MagickExport void SetBlobExempt(Image *image,const MagickBooleanType exempt) { assert(image != (const Image *) NULL); assert(image->signature == MagickCoreSignature); if (image->debug != MagickFalse) (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",image->filename); image->blob->exempt=exempt; } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + S e t B l o b E x t e n t % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % SetBlobExtent() ensures enough space is allocated for the blob. If the % method is successful, subsequent writes to bytes in the specified range are % guaranteed not to fail. % % The format of the SetBlobExtent method is: % % MagickBooleanType SetBlobExtent(Image *image,const MagickSizeType extent) % % A description of each parameter follows: % % o image: the image. % % o extent: the blob maximum extent. % */ MagickExport MagickBooleanType SetBlobExtent(Image *image, const MagickSizeType extent) { BlobInfo *magick_restrict blob_info; assert(image != (Image *) NULL); assert(image->signature == MagickCoreSignature); if (image->debug != MagickFalse) (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",image->filename); assert(image->blob != (BlobInfo *) NULL); assert(image->blob->type != UndefinedStream); blob_info=image->blob; switch (blob_info->type) { case UndefinedStream: break; case StandardStream: return(MagickFalse); case FileStream: { MagickOffsetType offset; ssize_t count; if (extent != (MagickSizeType) ((off_t) extent)) return(MagickFalse); offset=SeekBlob(image,0,SEEK_END); if (offset < 0) return(MagickFalse); if ((MagickSizeType) offset >= extent) break; offset=SeekBlob(image,(MagickOffsetType) extent-1,SEEK_SET); if (offset < 0) break; count=(ssize_t) fwrite((const unsigned char *) "",1,1, blob_info->file_info.file); #if defined(MAGICKCORE_HAVE_POSIX_FALLOCATE) if (blob_info->synchronize != MagickFalse) { int file; file=fileno(blob_info->file_info.file); if ((file == -1) || (offset < 0)) return(MagickFalse); (void) posix_fallocate(file,offset,extent-offset); } #endif offset=SeekBlob(image,offset,SEEK_SET); if (count != 1) return(MagickFalse); break; } case PipeStream: case ZipStream: return(MagickFalse); case BZipStream: return(MagickFalse); case FifoStream: return(MagickFalse); case BlobStream: { if (extent != (MagickSizeType) ((size_t) extent)) return(MagickFalse); if (blob_info->mapped != MagickFalse) { MagickOffsetType offset; ssize_t count; (void) UnmapBlob(blob_info->data,blob_info->length); RelinquishMagickResource(MapResource,blob_info->length); if (extent != (MagickSizeType) ((off_t) extent)) return(MagickFalse); offset=SeekBlob(image,0,SEEK_END); if (offset < 0) return(MagickFalse); if ((MagickSizeType) offset >= extent) break; offset=SeekBlob(image,(MagickOffsetType) extent-1,SEEK_SET); count=(ssize_t) fwrite((const unsigned char *) "",1,1, blob_info->file_info.file); #if defined(MAGICKCORE_HAVE_POSIX_FALLOCATE) if (blob_info->synchronize != MagickFalse) { int file; file=fileno(blob_info->file_info.file); if ((file == -1) || (offset < 0)) return(MagickFalse); (void) posix_fallocate(file,offset,extent-offset); } #endif offset=SeekBlob(image,offset,SEEK_SET); if (count != 1) return(MagickFalse); (void) AcquireMagickResource(MapResource,extent); blob_info->data=(unsigned char*) MapBlob(fileno( blob_info->file_info.file),WriteMode,0,(size_t) extent); blob_info->extent=(size_t) extent; blob_info->length=(size_t) extent; (void) SyncBlob(image); break; } blob_info->extent=(size_t) extent; blob_info->data=(unsigned char *) ResizeQuantumMemory(blob_info->data, blob_info->extent+1,sizeof(*blob_info->data)); (void) SyncBlob(image); if (blob_info->data == (unsigned char *) NULL) { (void) DetachBlob(blob_info); return(MagickFalse); } break; } case CustomStream: break; } return(MagickTrue); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + S e t C u s t o m S t r e a m D a t a % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % SetCustomStreamData() sets the stream info data member. % % The format of the SetCustomStreamData method is: % % void SetCustomStreamData(CustomStreamInfo *custom_stream,void *) % % A description of each parameter follows: % % o custom_stream: the custom stream info. % % o data: an object containing information about the custom stream. % */ MagickExport void SetCustomStreamData(CustomStreamInfo *custom_stream, void *data) { assert(custom_stream != (CustomStreamInfo *) NULL); assert(custom_stream->signature == MagickCoreSignature); custom_stream->data=data; } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + S e t C u s t o m S t r e a m R e a d e r % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % SetCustomStreamReader() sets the stream info reader member. % % The format of the SetCustomStreamReader method is: % % void SetCustomStreamReader(CustomStreamInfo *custom_stream, % CustomStreamHandler reader) % % A description of each parameter follows: % % o custom_stream: the custom stream info. % % o reader: a function to read from the stream. % */ MagickExport void SetCustomStreamReader(CustomStreamInfo *custom_stream, CustomStreamHandler reader) { assert(custom_stream != (CustomStreamInfo *) NULL); assert(custom_stream->signature == MagickCoreSignature); custom_stream->reader=reader; } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + S e t C u s t o m S t r e a m S e e k e r % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % SetCustomStreamSeeker() sets the stream info seeker member. % % The format of the SetCustomStreamReader method is: % % void SetCustomStreamSeeker(CustomStreamInfo *custom_stream, % CustomStreamSeeker seeker) % % A description of each parameter follows: % % o custom_stream: the custom stream info. % % o seeker: a function to seek in the custom stream. % */ MagickExport void SetCustomStreamSeeker(CustomStreamInfo *custom_stream, CustomStreamSeeker seeker) { assert(custom_stream != (CustomStreamInfo *) NULL); assert(custom_stream->signature == MagickCoreSignature); custom_stream->seeker=seeker; } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + S e t C u s t o m S t r e a m T e l l e r % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % SetCustomStreamTeller() sets the stream info teller member. % % The format of the SetCustomStreamTeller method is: % % void SetCustomStreamTeller(CustomStreamInfo *custom_stream, % CustomStreamTeller *teller) % % A description of each parameter follows: % % o custom_stream: the custom stream info. % % o teller: a function to set the position in the stream. % */ MagickExport void SetCustomStreamTeller(CustomStreamInfo *custom_stream, CustomStreamTeller teller) { assert(custom_stream != (CustomStreamInfo *) NULL); assert(custom_stream->signature == MagickCoreSignature); custom_stream->teller=teller; } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + S e t C u s t o m S t r e a m W r i t e r % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % SetCustomStreamWriter() sets the stream info writer member. % % The format of the SetCustomStreamWriter method is: % % void SetCustomStreamWriter(CustomStreamInfo *custom_stream, % CustomStreamHandler *writer) % % A description of each parameter follows: % % o custom_stream: the custom stream info. % % o writer: a function to write to the custom stream. % */ MagickExport void SetCustomStreamWriter(CustomStreamInfo *custom_stream, CustomStreamHandler writer) { assert(custom_stream != (CustomStreamInfo *) NULL); assert(custom_stream->signature == MagickCoreSignature); custom_stream->writer=writer; } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + S y n c B l o b % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % SyncBlob() flushes the datastream if it is a file or synchronizes the data % attributes if it is an blob. % % The format of the SyncBlob method is: % % int SyncBlob(Image *image) % % A description of each parameter follows: % % o image: the image. % */ static int SyncBlob(Image *image) { BlobInfo *magick_restrict blob_info; int status; assert(image != (Image *) NULL); assert(image->signature == MagickCoreSignature); if (image->debug != MagickFalse) (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",image->filename); assert(image->blob != (BlobInfo *) NULL); assert(image->blob->type != UndefinedStream); blob_info=image->blob; status=0; switch (blob_info->type) { case UndefinedStream: case StandardStream: break; case FileStream: case PipeStream: { status=fflush(blob_info->file_info.file); break; } case ZipStream: { #if defined(MAGICKCORE_ZLIB_DELEGATE) status=gzflush(blob_info->file_info.gzfile,Z_SYNC_FLUSH); #endif break; } case BZipStream: { #if defined(MAGICKCORE_BZLIB_DELEGATE) status=BZ2_bzflush(blob_info->file_info.bzfile); #endif break; } case FifoStream: break; case BlobStream: break; case CustomStream: break; } return(status); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + T e l l B l o b % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % TellBlob() obtains the current value of the blob or file position. % % The format of the TellBlob method is: % % MagickOffsetType TellBlob(const Image *image) % % A description of each parameter follows: % % o image: the image. % */ MagickExport MagickOffsetType TellBlob(const Image *image) { BlobInfo *magick_restrict blob_info; MagickOffsetType offset; assert(image != (Image *) NULL); assert(image->signature == MagickCoreSignature); if (image->debug != MagickFalse) (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",image->filename); assert(image->blob != (BlobInfo *) NULL); assert(image->blob->type != UndefinedStream); blob_info=image->blob; offset=(-1); switch (blob_info->type) { case UndefinedStream: case StandardStream: break; case FileStream: { offset=ftell(blob_info->file_info.file); break; } case PipeStream: break; case ZipStream: { #if defined(MAGICKCORE_ZLIB_DELEGATE) offset=(MagickOffsetType) gztell(blob_info->file_info.gzfile); #endif break; } case BZipStream: break; case FifoStream: break; case BlobStream: { offset=blob_info->offset; break; } case CustomStream: { if (blob_info->custom_stream->teller != (CustomStreamTeller) NULL) offset=blob_info->custom_stream->teller(blob_info->custom_stream->data); break; } } return(offset); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + U n m a p B l o b % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % UnmapBlob() deallocates the binary large object previously allocated with % the MapBlob method. % % The format of the UnmapBlob method is: % % MagickBooleanType UnmapBlob(void *map,const size_t length) % % A description of each parameter follows: % % o map: the address of the binary large object. % % o length: the length of the binary large object. % */ MagickExport MagickBooleanType UnmapBlob(void *map,const size_t length) { #if defined(MAGICKCORE_HAVE_MMAP) int status; status=munmap(map,length); return(status == -1 ? MagickFalse : MagickTrue); #else (void) map; (void) length; return(MagickFalse); #endif } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + W r i t e B l o b % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % WriteBlob() writes data to a blob or image file. It returns the number of % bytes written. % % The format of the WriteBlob method is: % % ssize_t WriteBlob(Image *image,const size_t length,const void *data) % % A description of each parameter follows: % % o image: the image. % % o length: Specifies an integer representing the number of bytes to % write to the file. % % o data: The address of the data to write to the blob or file. % */ MagickExport ssize_t WriteBlob(Image *image,const size_t length, const void *data) { BlobInfo *magick_restrict blob_info; int c; const unsigned char *p; unsigned char *q; ssize_t count; assert(image != (Image *) NULL); assert(image->signature == MagickCoreSignature); assert(image->blob != (BlobInfo *) NULL); assert(image->blob->type != UndefinedStream); if (length == 0) return(0); assert(data != (const void *) NULL); blob_info=image->blob; count=0; p=(const unsigned char *) data; q=(unsigned char *) data; switch (blob_info->type) { case UndefinedStream: break; case StandardStream: case FileStream: case PipeStream: { switch (length) { default: { count=(ssize_t) fwrite((const char *) data,1,length, blob_info->file_info.file); break; } case 4: { c=putc((int) *p++,blob_info->file_info.file); if (c == EOF) break; count++; } case 3: { c=putc((int) *p++,blob_info->file_info.file); if (c == EOF) break; count++; } case 2: { c=putc((int) *p++,blob_info->file_info.file); if (c == EOF) break; count++; } case 1: { c=putc((int) *p++,blob_info->file_info.file); if (c == EOF) break; count++; } case 0: break; } if ((count != (ssize_t) length) && (ferror(blob_info->file_info.file) != 0)) ThrowBlobException(blob_info); break; } case ZipStream: { #if defined(MAGICKCORE_ZLIB_DELEGATE) int status; switch (length) { default: { ssize_t i; for (i=0; i < (ssize_t) length; i+=count) { count=(ssize_t) gzwrite(blob_info->file_info.gzfile,q+i, (unsigned int) MagickMin(length-i,MagickMaxBufferExtent)); if (count <= 0) { count=0; if (errno != EINTR) break; } } count=i; break; } case 4: { c=gzputc(blob_info->file_info.gzfile,(int) *p++); if (c == EOF) break; count++; } case 3: { c=gzputc(blob_info->file_info.gzfile,(int) *p++); if (c == EOF) break; count++; } case 2: { c=gzputc(blob_info->file_info.gzfile,(int) *p++); if (c == EOF) break; count++; } case 1: { c=gzputc(blob_info->file_info.gzfile,(int) *p++); if (c == EOF) break; count++; } case 0: break; } status=Z_OK; (void) gzerror(blob_info->file_info.gzfile,&status); if ((count != (ssize_t) length) && (status != Z_OK)) ThrowBlobException(blob_info); #endif break; } case BZipStream: { #if defined(MAGICKCORE_BZLIB_DELEGATE) int status; ssize_t i; for (i=0; i < (ssize_t) length; i+=count) { count=(ssize_t) BZ2_bzwrite(blob_info->file_info.bzfile,q+i, (int) MagickMin(length-i,MagickMaxBufferExtent)); if (count <= 0) { count=0; if (errno != EINTR) break; } } count=i; status=BZ_OK; (void) BZ2_bzerror(blob_info->file_info.bzfile,&status); if ((count != (ssize_t) length) && (status != BZ_OK)) ThrowBlobException(blob_info); #endif break; } case FifoStream: { count=(ssize_t) blob_info->stream(image,data,length); break; } case BlobStream: { if ((blob_info->offset+(MagickOffsetType) length) >= (MagickOffsetType) blob_info->extent) { if (blob_info->mapped != MagickFalse) return(0); blob_info->extent+=length+blob_info->quantum; blob_info->quantum<<=1; blob_info->data=(unsigned char *) ResizeQuantumMemory( blob_info->data,blob_info->extent+1,sizeof(*blob_info->data)); (void) SyncBlob(image); if (blob_info->data == (unsigned char *) NULL) { (void) DetachBlob(blob_info); return(0); } } q=blob_info->data+blob_info->offset; (void) memcpy(q,p,length); blob_info->offset+=length; if (blob_info->offset >= (MagickOffsetType) blob_info->length) blob_info->length=(size_t) blob_info->offset; count=(ssize_t) length; break; } case CustomStream: { if (blob_info->custom_stream->writer != (CustomStreamHandler) NULL) count=blob_info->custom_stream->writer((unsigned char *) data, length,blob_info->custom_stream->data); break; } } return(count); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + W r i t e B l o b B y t e % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % WriteBlobByte() write an integer to a blob. It returns the number of bytes % written (either 0 or 1); % % The format of the WriteBlobByte method is: % % ssize_t WriteBlobByte(Image *image,const unsigned char value) % % A description of each parameter follows. % % o image: the image. % % o value: Specifies the value to write. % */ MagickExport ssize_t WriteBlobByte(Image *image,const unsigned char value) { BlobInfo *magick_restrict blob_info; ssize_t count; assert(image != (Image *) NULL); assert(image->signature == MagickCoreSignature); assert(image->blob != (BlobInfo *) NULL); assert(image->blob->type != UndefinedStream); blob_info=image->blob; count=0; switch (blob_info->type) { case StandardStream: case FileStream: case PipeStream: { int c; c=putc((int) value,blob_info->file_info.file); if (c == EOF) { if (ferror(blob_info->file_info.file) != 0) ThrowBlobException(blob_info); break; } count++; break; } default: { count=WriteBlobStream(image,1,&value); break; } } return(count); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + W r i t e B l o b F l o a t % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % WriteBlobFloat() writes a float value as a 32-bit quantity in the byte-order % specified by the endian member of the image structure. % % The format of the WriteBlobFloat method is: % % ssize_t WriteBlobFloat(Image *image,const float value) % % A description of each parameter follows. % % o image: the image. % % o value: Specifies the value to write. % */ MagickExport ssize_t WriteBlobFloat(Image *image,const float value) { union { unsigned int unsigned_value; float float_value; } quantum; quantum.unsigned_value=0U; quantum.float_value=value; return(WriteBlobLong(image,quantum.unsigned_value)); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + W r i t e B l o b L o n g % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % WriteBlobLong() writes a unsigned int value as a 32-bit quantity in the % byte-order specified by the endian member of the image structure. % % The format of the WriteBlobLong method is: % % ssize_t WriteBlobLong(Image *image,const unsigned int value) % % A description of each parameter follows. % % o image: the image. % % o value: Specifies the value to write. % */ MagickExport ssize_t WriteBlobLong(Image *image,const unsigned int value) { unsigned char buffer[4]; assert(image != (Image *) NULL); assert(image->signature == MagickCoreSignature); if (image->endian == LSBEndian) { buffer[0]=(unsigned char) value; buffer[1]=(unsigned char) (value >> 8); buffer[2]=(unsigned char) (value >> 16); buffer[3]=(unsigned char) (value >> 24); return(WriteBlobStream(image,4,buffer)); } buffer[0]=(unsigned char) (value >> 24); buffer[1]=(unsigned char) (value >> 16); buffer[2]=(unsigned char) (value >> 8); buffer[3]=(unsigned char) value; return(WriteBlobStream(image,4,buffer)); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + W r i t e B l o b L o n g L o n g % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % WriteBlobMSBLongLong() writes a long long value as a 64-bit quantity in the % byte-order specified by the endian member of the image structure. % % The format of the WriteBlobLongLong method is: % % ssize_t WriteBlobLongLong(Image *image,const MagickSizeType value) % % A description of each parameter follows. % % o value: Specifies the value to write. % % o image: the image. % */ MagickExport ssize_t WriteBlobLongLong(Image *image,const MagickSizeType value) { unsigned char buffer[8]; assert(image != (Image *) NULL); assert(image->signature == MagickCoreSignature); if (image->endian == LSBEndian) { buffer[0]=(unsigned char) value; buffer[1]=(unsigned char) (value >> 8); buffer[2]=(unsigned char) (value >> 16); buffer[3]=(unsigned char) (value >> 24); buffer[4]=(unsigned char) (value >> 32); buffer[5]=(unsigned char) (value >> 40); buffer[6]=(unsigned char) (value >> 48); buffer[7]=(unsigned char) (value >> 56); return(WriteBlobStream(image,8,buffer)); } buffer[0]=(unsigned char) (value >> 56); buffer[1]=(unsigned char) (value >> 48); buffer[2]=(unsigned char) (value >> 40); buffer[3]=(unsigned char) (value >> 32); buffer[4]=(unsigned char) (value >> 24); buffer[5]=(unsigned char) (value >> 16); buffer[6]=(unsigned char) (value >> 8); buffer[7]=(unsigned char) value; return(WriteBlobStream(image,8,buffer)); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + W r i t e B l o b S h o r t % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % WriteBlobShort() writes a short value as a 16-bit quantity in the % byte-order specified by the endian member of the image structure. % % The format of the WriteBlobShort method is: % % ssize_t WriteBlobShort(Image *image,const unsigned short value) % % A description of each parameter follows. % % o image: the image. % % o value: Specifies the value to write. % */ MagickExport ssize_t WriteBlobShort(Image *image,const unsigned short value) { unsigned char buffer[2]; assert(image != (Image *) NULL); assert(image->signature == MagickCoreSignature); if (image->endian == LSBEndian) { buffer[0]=(unsigned char) value; buffer[1]=(unsigned char) (value >> 8); return(WriteBlobStream(image,2,buffer)); } buffer[0]=(unsigned char) (value >> 8); buffer[1]=(unsigned char) value; return(WriteBlobStream(image,2,buffer)); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + W r i t e B l o b S i g n e d L o n g % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % WriteBlobSignedLong() writes a signed value as a 32-bit quantity in the % byte-order specified by the endian member of the image structure. % % The format of the WriteBlobSignedLong method is: % % ssize_t WriteBlobSignedLong(Image *image,const signed int value) % % A description of each parameter follows. % % o image: the image. % % o value: Specifies the value to write. % */ MagickExport ssize_t WriteBlobSignedLong(Image *image,const signed int value) { union { unsigned int unsigned_value; signed int signed_value; } quantum; unsigned char buffer[4]; assert(image != (Image *) NULL); assert(image->signature == MagickCoreSignature); quantum.signed_value=value; if (image->endian == LSBEndian) { buffer[0]=(unsigned char) quantum.unsigned_value; buffer[1]=(unsigned char) (quantum.unsigned_value >> 8); buffer[2]=(unsigned char) (quantum.unsigned_value >> 16); buffer[3]=(unsigned char) (quantum.unsigned_value >> 24); return(WriteBlobStream(image,4,buffer)); } buffer[0]=(unsigned char) (quantum.unsigned_value >> 24); buffer[1]=(unsigned char) (quantum.unsigned_value >> 16); buffer[2]=(unsigned char) (quantum.unsigned_value >> 8); buffer[3]=(unsigned char) quantum.unsigned_value; return(WriteBlobStream(image,4,buffer)); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + W r i t e B l o b L S B L o n g % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % WriteBlobLSBLong() writes a unsigned int value as a 32-bit quantity in % least-significant byte first order. % % The format of the WriteBlobLSBLong method is: % % ssize_t WriteBlobLSBLong(Image *image,const unsigned int value) % % A description of each parameter follows. % % o image: the image. % % o value: Specifies the value to write. % */ MagickExport ssize_t WriteBlobLSBLong(Image *image,const unsigned int value) { unsigned char buffer[4]; assert(image != (Image *) NULL); assert(image->signature == MagickCoreSignature); buffer[0]=(unsigned char) value; buffer[1]=(unsigned char) (value >> 8); buffer[2]=(unsigned char) (value >> 16); buffer[3]=(unsigned char) (value >> 24); return(WriteBlobStream(image,4,buffer)); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + W r i t e B l o b L S B S h o r t % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % WriteBlobLSBShort() writes a unsigned short value as a 16-bit quantity in % least-significant byte first order. % % The format of the WriteBlobLSBShort method is: % % ssize_t WriteBlobLSBShort(Image *image,const unsigned short value) % % A description of each parameter follows. % % o image: the image. % % o value: Specifies the value to write. % */ MagickExport ssize_t WriteBlobLSBShort(Image *image,const unsigned short value) { unsigned char buffer[2]; assert(image != (Image *) NULL); assert(image->signature == MagickCoreSignature); buffer[0]=(unsigned char) value; buffer[1]=(unsigned char) (value >> 8); return(WriteBlobStream(image,2,buffer)); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + W r i t e B l o b L S B S i g n e d L o n g % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % WriteBlobLSBSignedLong() writes a signed value as a 32-bit quantity in % least-significant byte first order. % % The format of the WriteBlobLSBSignedLong method is: % % ssize_t WriteBlobLSBSignedLong(Image *image,const signed int value) % % A description of each parameter follows. % % o image: the image. % % o value: Specifies the value to write. % */ MagickExport ssize_t WriteBlobLSBSignedLong(Image *image,const signed int value) { union { unsigned int unsigned_value; signed int signed_value; } quantum; unsigned char buffer[4]; assert(image != (Image *) NULL); assert(image->signature == MagickCoreSignature); quantum.signed_value=value; buffer[0]=(unsigned char) quantum.unsigned_value; buffer[1]=(unsigned char) (quantum.unsigned_value >> 8); buffer[2]=(unsigned char) (quantum.unsigned_value >> 16); buffer[3]=(unsigned char) (quantum.unsigned_value >> 24); return(WriteBlobStream(image,4,buffer)); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + W r i t e B l o b L S B S i g n e d S h o r t % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % WriteBlobLSBSignedShort() writes a signed short value as a 16-bit quantity % in least-significant byte first order. % % The format of the WriteBlobLSBSignedShort method is: % % ssize_t WriteBlobLSBSignedShort(Image *image,const signed short value) % % A description of each parameter follows. % % o image: the image. % % o value: Specifies the value to write. % */ MagickExport ssize_t WriteBlobLSBSignedShort(Image *image, const signed short value) { union { unsigned short unsigned_value; signed short signed_value; } quantum; unsigned char buffer[2]; assert(image != (Image *) NULL); assert(image->signature == MagickCoreSignature); quantum.signed_value=value; buffer[0]=(unsigned char) quantum.unsigned_value; buffer[1]=(unsigned char) (quantum.unsigned_value >> 8); return(WriteBlobStream(image,2,buffer)); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + W r i t e B l o b M S B L o n g % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % WriteBlobMSBLong() writes a unsigned int value as a 32-bit quantity in % most-significant byte first order. % % The format of the WriteBlobMSBLong method is: % % ssize_t WriteBlobMSBLong(Image *image,const unsigned int value) % % A description of each parameter follows. % % o value: Specifies the value to write. % % o image: the image. % */ MagickExport ssize_t WriteBlobMSBLong(Image *image,const unsigned int value) { unsigned char buffer[4]; assert(image != (Image *) NULL); assert(image->signature == MagickCoreSignature); buffer[0]=(unsigned char) (value >> 24); buffer[1]=(unsigned char) (value >> 16); buffer[2]=(unsigned char) (value >> 8); buffer[3]=(unsigned char) value; return(WriteBlobStream(image,4,buffer)); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + W r i t e B l o b M S B S i g n e d S h o r t % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % WriteBlobMSBSignedShort() writes a signed short value as a 16-bit quantity % in most-significant byte first order. % % The format of the WriteBlobMSBSignedShort method is: % % ssize_t WriteBlobMSBSignedShort(Image *image,const signed short value) % % A description of each parameter follows. % % o image: the image. % % o value: Specifies the value to write. % */ MagickExport ssize_t WriteBlobMSBSignedShort(Image *image, const signed short value) { union { unsigned short unsigned_value; signed short signed_value; } quantum; unsigned char buffer[2]; assert(image != (Image *) NULL); assert(image->signature == MagickCoreSignature); quantum.signed_value=value; buffer[0]=(unsigned char) (quantum.unsigned_value >> 8); buffer[1]=(unsigned char) quantum.unsigned_value; return(WriteBlobStream(image,2,buffer)); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + W r i t e B l o b M S B S h o r t % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % WriteBlobMSBShort() writes a unsigned short value as a 16-bit quantity in % most-significant byte first order. % % The format of the WriteBlobMSBShort method is: % % ssize_t WriteBlobMSBShort(Image *image,const unsigned short value) % % A description of each parameter follows. % % o value: Specifies the value to write. % % o file: Specifies the file to write the data to. % */ MagickExport ssize_t WriteBlobMSBShort(Image *image,const unsigned short value) { unsigned char buffer[2]; assert(image != (Image *) NULL); assert(image->signature == MagickCoreSignature); buffer[0]=(unsigned char) (value >> 8); buffer[1]=(unsigned char) value; return(WriteBlobStream(image,2,buffer)); } /* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % % % % % + W r i t e B l o b S t r i n g % % % % % % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % WriteBlobString() write a string to a blob. It returns the number of % characters written. % % The format of the WriteBlobString method is: % % ssize_t WriteBlobString(Image *image,const char *string) % % A description of each parameter follows. % % o image: the image. % % o string: Specifies the string to write. % */ MagickExport ssize_t WriteBlobString(Image *image,const char *string) { assert(image != (Image *) NULL); assert(image->signature == MagickCoreSignature); assert(string != (const char *) NULL); return(WriteBlobStream(image,strlen(string),(const unsigned char *) string)); }