This source file includes following definitions.
- dpc_read
- ConnectPixelCacheServer
- GetHostname
- AcquireDistributeCacheInfo
- DestroyDistributeCacheInfo
- DestroyDistributeCache
- dpc_send
- OpenDistributeCache
- ReadDistributeCacheIndexes
- ReadDistributeCachePixels
- RelinquishImageRegistry
- WriteDistributeCacheIndexes
- WriteDistributeCachePixels
- DistributePixelCacheClient
- DistributePixelCacheServer
- GetDistributeCacheFile
- GetDistributeCacheHostname
- GetDistributeCachePort
- OpenDistributePixelCache
- ReadDistributePixelCacheIndexes
- ReadDistributePixelCachePixels
- RelinquishDistributePixelCache
- WriteDistributePixelCacheIndexes
- WriteDistributePixelCachePixels
#include "magick/studio.h"
#include "magick/cache.h"
#include "magick/cache-private.h"
#include "magick/distribute-cache.h"
#include "magick/distribute-cache-private.h"
#include "magick/exception.h"
#include "magick/exception-private.h"
#include "magick/geometry.h"
#include "magick/image.h"
#include "magick/image-private.h"
#include "magick/list.h"
#include "magick/locale_.h"
#include "magick/memory_.h"
#include "magick/nt-base-private.h"
#include "magick/policy.h"
#include "magick/random_.h"
#include "magick/registry.h"
#include "magick/splay-tree.h"
#include "magick/string_.h"
#include "magick/string-private.h"
#include "magick/version.h"
#include "magick/version-private.h"
#undef MAGICKCORE_HAVE_DISTRIBUTE_CACHE
#if defined(MAGICKCORE_HAVE_SOCKET) && defined(MAGICKCORE_THREAD_SUPPORT)
#include <netinet/in.h>
#include <netdb.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#define CHAR_TYPE_CAST
#define CLOSE_SOCKET(socket) (void) close(socket)
#define HANDLER_RETURN_TYPE void *
#define HANDLER_RETURN_VALUE (void *) NULL
#define SOCKET_TYPE int
#define LENGTH_TYPE size_t
#define MAGICKCORE_HAVE_DISTRIBUTE_CACHE
#elif defined(MAGICKCORE_WINDOWS_SUPPORT)
#define CHAR_TYPE_CAST (char *)
#define CLOSE_SOCKET(socket) (void) closesocket(socket)
#define HANDLER_RETURN_TYPE DWORD WINAPI
#define HANDLER_RETURN_VALUE 0
#define SOCKET_TYPE SOCKET
#define LENGTH_TYPE int
#define MAGICKCORE_HAVE_DISTRIBUTE_CACHE
#else
#ifdef __VMS
#define CLOSE_SOCKET(socket) (void) close(socket)
#else
#define CLOSE_SOCKET(socket)
#endif
#define HANDLER_RETURN_TYPE void *
#define HANDLER_RETURN_VALUE (void *) NULL
#define SOCKET_TYPE int
#define LENGTH_TYPE size_t
#undef send
#undef recv
#define send(file,buffer,length,flags) 0
#define recv(file,buffer,length,flags) 0
#endif
#define DPCHostname "127.0.0.1"
#define DPCPendingConnections 10
#define DPCPort 6668
#define DPCSessionKeyLength 8
#ifndef MSG_NOSIGNAL
# define MSG_NOSIGNAL 0
#endif
static inline MagickOffsetType dpc_read(int file,const MagickSizeType length,
unsigned char *magick_restrict message)
{
register MagickOffsetType
i;
ssize_t
count;
#if !defined(MAGICKCORE_HAVE_DISTRIBUTE_CACHE)
magick_unreferenced(file);
magick_unreferenced(message);
#endif
count=0;
for (i=0; i < (MagickOffsetType) length; i+=count)
{
count=recv(file,CHAR_TYPE_CAST message+i,(LENGTH_TYPE) MagickMin(length-i,
(MagickSizeType) SSIZE_MAX),0);
if (count <= 0)
{
count=0;
if (errno != EINTR)
break;
}
}
return(i);
}
static int ConnectPixelCacheServer(const char *hostname,const int port,
size_t *session_key,ExceptionInfo *exception)
{
#if defined(MAGICKCORE_HAVE_DISTRIBUTE_CACHE)
char
service[MaxTextExtent];
const char
*shared_secret;
int
status;
SOCKET_TYPE
client_socket;
ssize_t
count;
struct addrinfo
hint,
*result;
unsigned char
secret[MaxTextExtent];
*session_key=0;
shared_secret=GetPolicyValue("shared-secret");
if (shared_secret == (const char *) NULL)
{
(void) ThrowMagickException(exception,GetMagickModule(),CacheError,
"DistributedPixelCache","'%s'","shared secret expected");
return(-1);
}
#if defined(MAGICKCORE_WINDOWS_SUPPORT)
NTInitializeWinsock(MagickTrue);
#endif
(void) ResetMagickMemory(&hint,0,sizeof(hint));
hint.ai_family=AF_INET;
hint.ai_socktype=SOCK_STREAM;
hint.ai_flags=AI_PASSIVE;
(void) FormatLocaleString(service,MaxTextExtent,"%d",port);
status=getaddrinfo(hostname,service,&hint,&result);
if (status != 0)
{
(void) ThrowMagickException(exception,GetMagickModule(),CacheError,
"DistributedPixelCache","'%s'",hostname);
return(-1);
}
client_socket=socket(result->ai_family,result->ai_socktype,
result->ai_protocol);
if (client_socket == -1)
{
freeaddrinfo(result);
(void) ThrowMagickException(exception,GetMagickModule(),CacheError,
"DistributedPixelCache","'%s'",hostname);
return(-1);
}
status=connect(client_socket,result->ai_addr,(socklen_t) result->ai_addrlen);
if (status == -1)
{
CLOSE_SOCKET(client_socket);
freeaddrinfo(result);
(void) ThrowMagickException(exception,GetMagickModule(),CacheError,
"DistributedPixelCache","'%s'",hostname);
return(-1);
}
count=recv(client_socket,CHAR_TYPE_CAST secret,MaxTextExtent,0);
if (count != -1)
{
StringInfo
*nonce;
nonce=AcquireStringInfo(count);
(void) memcpy(GetStringInfoDatum(nonce),secret,(size_t) count);
*session_key=GetMagickSignature(nonce);
nonce=DestroyStringInfo(nonce);
}
if (*session_key == 0)
{
CLOSE_SOCKET(client_socket);
client_socket=(SOCKET_TYPE) (-1);
}
freeaddrinfo(result);
return(client_socket);
#else
magick_unreferenced(hostname);
magick_unreferenced(port);
magick_unreferenced(session_key);
magick_unreferenced(exception);
(void) ThrowMagickException(exception,GetMagickModule(),MissingDelegateError,
"DelegateLibrarySupportNotBuiltIn","distributed pixel cache");
return(MagickFalse);
#endif
}
static char *GetHostname(int *port,ExceptionInfo *exception)
{
char
*host,
*hosts,
**hostlist;
int
argc;
register ssize_t
i;
static size_t
id = 0;
hosts=(char *) GetImageRegistry(StringRegistryType,"cache:hosts",exception);
if (hosts == (char *) NULL)
{
*port=DPCPort;
return(AcquireString(DPCHostname));
}
(void) SubstituteString(&hosts,","," ");
hostlist=StringToArgv(hosts,&argc);
hosts=DestroyString(hosts);
if (hostlist == (char **) NULL)
{
*port=DPCPort;
return(AcquireString(DPCHostname));
}
hosts=AcquireString(hostlist[(id++ % (argc-1))+1]);
for (i=0; i < (ssize_t) argc; i++)
hostlist[i]=DestroyString(hostlist[i]);
hostlist=(char **) RelinquishMagickMemory(hostlist);
(void) SubstituteString(&hosts,":"," ");
hostlist=StringToArgv(hosts,&argc);
if (hostlist == (char **) NULL)
{
*port=DPCPort;
return(AcquireString(DPCHostname));
}
host=AcquireString(hostlist[1]);
if (hostlist[2] == (char *) NULL)
*port=DPCPort;
else
*port=StringToLong(hostlist[2]);
for (i=0; i < (ssize_t) argc; i++)
hostlist[i]=DestroyString(hostlist[i]);
hostlist=(char **) RelinquishMagickMemory(hostlist);
return(host);
}
MagickPrivate DistributeCacheInfo *AcquireDistributeCacheInfo(
ExceptionInfo *exception)
{
char
*hostname;
DistributeCacheInfo
*server_info;
size_t
session_key;
server_info=(DistributeCacheInfo *) AcquireMagickMemory(sizeof(*server_info));
if (server_info == (DistributeCacheInfo *) NULL)
ThrowFatalException(ResourceLimitFatalError,"MemoryAllocationFailed");
(void) ResetMagickMemory(server_info,0,sizeof(*server_info));
server_info->signature=MagickSignature;
server_info->port=0;
hostname=GetHostname(&server_info->port,exception);
session_key=0;
server_info->file=ConnectPixelCacheServer(hostname,server_info->port,
&session_key,exception);
if (server_info->file == -1)
server_info=DestroyDistributeCacheInfo(server_info);
else
{
server_info->session_key=session_key;
(void) CopyMagickString(server_info->hostname,hostname,MaxTextExtent);
server_info->debug=IsEventLogging();
}
hostname=DestroyString(hostname);
return(server_info);
}
MagickPrivate DistributeCacheInfo *DestroyDistributeCacheInfo(
DistributeCacheInfo *server_info)
{
assert(server_info != (DistributeCacheInfo *) NULL);
assert(server_info->signature == MagickSignature);
if (server_info->file > 0)
CLOSE_SOCKET(server_info->file);
server_info->signature=(~MagickSignature);
server_info=(DistributeCacheInfo *) RelinquishMagickMemory(server_info);
return(server_info);
}
static MagickBooleanType DestroyDistributeCache(SplayTreeInfo *registry,
const size_t session_key)
{
return(DeleteNodeFromSplayTree(registry,(const void *) session_key));
}
static inline MagickOffsetType dpc_send(int file,const MagickSizeType length,
const unsigned char *magick_restrict message)
{
MagickOffsetType
count;
register MagickOffsetType
i;
#if !defined(MAGICKCORE_HAVE_DISTRIBUTE_CACHE)
magick_unreferenced(file);
magick_unreferenced(message);
#endif
count=0;
for (i=0; i < (MagickOffsetType) length; i+=count)
{
count=(MagickOffsetType) send(file,CHAR_TYPE_CAST message+i,(LENGTH_TYPE)
MagickMin(length-i,(MagickSizeType) SSIZE_MAX),MSG_NOSIGNAL);
if (count <= 0)
{
count=0;
if (errno != EINTR)
break;
}
}
return(i);
}
static MagickBooleanType OpenDistributeCache(SplayTreeInfo *registry,int file,
const size_t session_key,ExceptionInfo *exception)
{
Image
*image;
MagickBooleanType
status;
MagickOffsetType
count;
MagickSizeType
length;
register unsigned char
*p;
unsigned char
message[MaxTextExtent];
image=AcquireImage((ImageInfo *) NULL);
if (image == (Image *) NULL)
return(MagickFalse);
length=sizeof(image->storage_class)+sizeof(image->colorspace)+
sizeof(image->channels)+sizeof(image->columns)+sizeof(image->rows);
count=dpc_read(file,length,message);
if (count != (MagickOffsetType) length)
return(MagickFalse);
p=message;
(void) memcpy(&image->storage_class,p,sizeof(image->storage_class));
p+=sizeof(image->storage_class);
(void) memcpy(&image->colorspace,p,sizeof(image->colorspace));
p+=sizeof(image->colorspace);
(void) memcpy(&image->channels,p,sizeof(image->channels));
p+=sizeof(image->channels);
(void) memcpy(&image->columns,p,sizeof(image->columns));
p+=sizeof(image->columns);
(void) memcpy(&image->rows,p,sizeof(image->rows));
p+=sizeof(image->rows);
if (SyncImagePixelCache(image,exception) == MagickFalse)
return(MagickFalse);
status=AddValueToSplayTree(registry,(const void *) session_key,image);
return(status);
}
static MagickBooleanType ReadDistributeCacheIndexes(SplayTreeInfo *registry,
int file,const size_t session_key,ExceptionInfo *exception)
{
const IndexPacket
*indexes;
Image
*image;
MagickOffsetType
count;
MagickSizeType
length;
RectangleInfo
region;
register const PixelPacket
*p;
register unsigned char
*q;
unsigned char
message[MaxTextExtent];
image=(Image *) GetValueFromSplayTree(registry,(const void *) session_key);
if (image == (Image *) NULL)
return(MagickFalse);
length=sizeof(region.width)+sizeof(region.height)+sizeof(region.x)+
sizeof(region.y)+sizeof(length);
count=dpc_read(file,length,message);
if (count != (MagickOffsetType) length)
return(MagickFalse);
q=message;
(void) memcpy(®ion.width,q,sizeof(region.width));
q+=sizeof(region.width);
(void) memcpy(®ion.height,q,sizeof(region.height));
q+=sizeof(region.height);
(void) memcpy(®ion.x,q,sizeof(region.x));
q+=sizeof(region.x);
(void) memcpy(®ion.y,q,sizeof(region.y));
q+=sizeof(region.y);
(void) memcpy(&length,q,sizeof(length));
q+=sizeof(length);
p=GetVirtualPixels(image,region.x,region.y,region.width,region.height,
exception);
if (p == (const PixelPacket *) NULL)
return(MagickFalse);
indexes=GetVirtualIndexQueue(image);
count=dpc_send(file,length,(unsigned char *) indexes);
if (count != (MagickOffsetType) length)
return(MagickFalse);
return(MagickTrue);
}
static MagickBooleanType ReadDistributeCachePixels(SplayTreeInfo *registry,
int file,const size_t session_key,ExceptionInfo *exception)
{
Image
*image;
MagickOffsetType
count;
MagickSizeType
length;
RectangleInfo
region;
register const PixelPacket
*p;
register unsigned char
*q;
unsigned char
message[MaxTextExtent];
image=(Image *) GetValueFromSplayTree(registry,(const void *) session_key);
if (image == (Image *) NULL)
return(MagickFalse);
length=sizeof(region.width)+sizeof(region.height)+sizeof(region.x)+
sizeof(region.y)+sizeof(length);
count=dpc_read(file,length,message);
if (count != (MagickOffsetType) length)
return(MagickFalse);
q=message;
(void) memcpy(®ion.width,q,sizeof(region.width));
q+=sizeof(region.width);
(void) memcpy(®ion.height,q,sizeof(region.height));
q+=sizeof(region.height);
(void) memcpy(®ion.x,q,sizeof(region.x));
q+=sizeof(region.x);
(void) memcpy(®ion.y,q,sizeof(region.y));
q+=sizeof(region.y);
(void) memcpy(&length,q,sizeof(length));
q+=sizeof(length);
p=GetVirtualPixels(image,region.x,region.y,region.width,region.height,
exception);
if (p == (const PixelPacket *) NULL)
return(MagickFalse);
count=dpc_send(file,length,(unsigned char *) p);
if (count != (MagickOffsetType) length)
return(MagickFalse);
return(MagickTrue);
}
static void *RelinquishImageRegistry(void *image)
{
return((void *) DestroyImageList((Image *) image));
}
static MagickBooleanType WriteDistributeCacheIndexes(SplayTreeInfo *registry,
int file,const size_t session_key,ExceptionInfo *exception)
{
Image
*image;
IndexPacket
*indexes;
MagickOffsetType
count;
MagickSizeType
length;
RectangleInfo
region;
register unsigned char
*p;
register PixelPacket
*q;
unsigned char
message[MaxTextExtent];
image=(Image *) GetValueFromSplayTree(registry,(const void *) session_key);
if (image == (Image *) NULL)
return(MagickFalse);
length=sizeof(region.width)+sizeof(region.height)+sizeof(region.x)+
sizeof(region.y)+sizeof(length);
count=dpc_read(file,length,message);
if (count != (MagickOffsetType) length)
return(MagickFalse);
p=message;
(void) memcpy(®ion.width,p,sizeof(region.width));
p+=sizeof(region.width);
(void) memcpy(®ion.height,p,sizeof(region.height));
p+=sizeof(region.height);
(void) memcpy(®ion.x,p,sizeof(region.x));
p+=sizeof(region.x);
(void) memcpy(®ion.y,p,sizeof(region.y));
p+=sizeof(region.y);
(void) memcpy(&length,p,sizeof(length));
p+=sizeof(length);
q=GetAuthenticPixels(image,region.x,region.y,region.width,region.height,
exception);
if (q == (PixelPacket *) NULL)
return(MagickFalse);
indexes=GetAuthenticIndexQueue(image);
count=dpc_read(file,length,(unsigned char *) indexes);
if (count != (MagickOffsetType) length)
return(MagickFalse);
return(SyncAuthenticPixels(image,exception));
}
static MagickBooleanType WriteDistributeCachePixels(SplayTreeInfo *registry,
int file,const size_t session_key,ExceptionInfo *exception)
{
Image
*image;
MagickOffsetType
count;
MagickSizeType
length;
RectangleInfo
region;
register PixelPacket
*q;
register unsigned char
*p;
unsigned char
message[MaxTextExtent];
image=(Image *) GetValueFromSplayTree(registry,(const void *) session_key);
if (image == (Image *) NULL)
return(MagickFalse);
length=sizeof(region.width)+sizeof(region.height)+sizeof(region.x)+
sizeof(region.y)+sizeof(length);
count=dpc_read(file,length,message);
if (count != (MagickOffsetType) length)
return(MagickFalse);
p=message;
(void) memcpy(®ion.width,p,sizeof(region.width));
p+=sizeof(region.width);
(void) memcpy(®ion.height,p,sizeof(region.height));
p+=sizeof(region.height);
(void) memcpy(®ion.x,p,sizeof(region.x));
p+=sizeof(region.x);
(void) memcpy(®ion.y,p,sizeof(region.y));
p+=sizeof(region.y);
(void) memcpy(&length,p,sizeof(length));
p+=sizeof(length);
q=GetAuthenticPixels(image,region.x,region.y,region.width,region.height,
exception);
if (q == (PixelPacket *) NULL)
return(MagickFalse);
count=dpc_read(file,length,(unsigned char *) q);
if (count != (MagickOffsetType) length)
return(MagickFalse);
return(SyncAuthenticPixels(image,exception));
}
static HANDLER_RETURN_TYPE DistributePixelCacheClient(void *socket)
{
const char
*shared_secret;
ExceptionInfo
*exception;
MagickBooleanType
status;
MagickOffsetType
count;
RandomInfo
*random_info;
register unsigned char
*p;
size_t
key,
session_key;
SOCKET_TYPE
client_socket;
SplayTreeInfo
*registry;
StringInfo
*secret;
unsigned char
command,
session[2*MaxTextExtent];
shared_secret=GetPolicyValue("shared-secret");
if (shared_secret == (const char *) NULL)
ThrowFatalException(CacheFatalError,"shared secret expected");
p=session;
(void) CopyMagickString((char *) p,shared_secret,MaxTextExtent);
p+=strlen(shared_secret);
random_info=AcquireRandomInfo();
secret=GetRandomKey(random_info,DPCSessionKeyLength);
(void) memcpy(p,GetStringInfoDatum(secret),DPCSessionKeyLength);
session_key=GetMagickSignature(secret);
random_info=DestroyRandomInfo(random_info);
exception=AcquireExceptionInfo();
registry=NewSplayTree((int (*)(const void *,const void *)) NULL,
(void *(*)(void *)) NULL,RelinquishImageRegistry);
client_socket=(*(int *) socket);
count=dpc_send(client_socket,DPCSessionKeyLength,GetStringInfoDatum(secret));
secret=DestroyStringInfo(secret);
for ( ; ; )
{
count=dpc_read(client_socket,1,(unsigned char *) &command);
if (count <= 0)
break;
count=dpc_read(client_socket,sizeof(key),(unsigned char *) &key);
if ((count != (MagickOffsetType) sizeof(key)) || (key != session_key))
break;
status=MagickFalse;
switch (command)
{
case 'o':
{
status=OpenDistributeCache(registry,client_socket,session_key,
exception);
count=dpc_send(client_socket,sizeof(status),(unsigned char *) &status);
break;
}
case 'r':
{
status=ReadDistributeCachePixels(registry,client_socket,session_key,
exception);
break;
}
case 'R':
{
status=ReadDistributeCacheIndexes(registry,client_socket,session_key,
exception);
break;
}
case 'w':
{
status=WriteDistributeCachePixels(registry,client_socket,session_key,
exception);
break;
}
case 'W':
{
status=WriteDistributeCacheIndexes(registry,client_socket,session_key,
exception);
break;
}
case 'd':
{
status=DestroyDistributeCache(registry,session_key);
break;
}
default:
break;
}
if (status == MagickFalse)
break;
if (command == 'd')
break;
}
count=dpc_send(client_socket,sizeof(status),(unsigned char *) &status);
CLOSE_SOCKET(client_socket);
exception=DestroyExceptionInfo(exception);
registry=DestroySplayTree(registry);
return(HANDLER_RETURN_VALUE);
}
MagickExport void DistributePixelCacheServer(const int port,
ExceptionInfo *exception)
{
#if defined(MAGICKCORE_HAVE_DISTRIBUTE_CACHE)
char
service[MaxTextExtent];
int
status;
#if defined(MAGICKCORE_THREAD_SUPPORT)
pthread_attr_t
attributes;
pthread_t
threads;
#elif defined(MAGICKCORE_WINDOWS_SUPPORT)
DWORD
threadID;
#else
Not implemented!
#endif
register struct addrinfo
*p;
SOCKET_TYPE
server_socket;
struct addrinfo
hint,
*result;
struct sockaddr_in
address;
magick_unreferenced(exception);
#if defined(MAGICKCORE_WINDOWS_SUPPORT)
NTInitializeWinsock(MagickFalse);
#endif
(void) ResetMagickMemory(&hint,0,sizeof(hint));
hint.ai_family=AF_INET;
hint.ai_socktype=SOCK_STREAM;
hint.ai_flags=AI_PASSIVE;
(void) FormatLocaleString(service,MaxTextExtent,"%d",port);
status=getaddrinfo((const char *) NULL,service,&hint,&result);
if (status != 0)
ThrowFatalException(CacheFatalError,"UnableToListen");
server_socket=(SOCKET_TYPE) 0;
for (p=result; p != (struct addrinfo *) NULL; p=p->ai_next)
{
int
one;
server_socket=socket(p->ai_family,p->ai_socktype,p->ai_protocol);
if (server_socket == -1)
continue;
one=1;
status=setsockopt(server_socket,SOL_SOCKET,SO_REUSEADDR,
CHAR_TYPE_CAST &one,(socklen_t) sizeof(one));
if (status == -1)
{
CLOSE_SOCKET(server_socket);
continue;
}
status=bind(server_socket,p->ai_addr,(socklen_t) p->ai_addrlen);
if (status == -1)
{
CLOSE_SOCKET(server_socket);
continue;
}
break;
}
if (p == (struct addrinfo *) NULL)
ThrowFatalException(CacheFatalError,"UnableToBind");
freeaddrinfo(result);
status=listen(server_socket,DPCPendingConnections);
if (status != 0)
ThrowFatalException(CacheFatalError,"UnableToListen");
#if defined(MAGICKCORE_THREAD_SUPPORT)
pthread_attr_init(&attributes);
#endif
for ( ; ; )
{
SOCKET_TYPE
client_socket;
socklen_t
length;
length=(socklen_t) sizeof(address);
client_socket=accept(server_socket,(struct sockaddr *) &address,&length);
if (client_socket == -1)
ThrowFatalException(CacheFatalError,"UnableToEstablishConnection");
#if defined(MAGICKCORE_THREAD_SUPPORT)
status=pthread_create(&threads,&attributes,DistributePixelCacheClient,
(void *) &client_socket);
if (status == -1)
ThrowFatalException(CacheFatalError,"UnableToCreateClientThread");
#elif defined(MAGICKCORE_WINDOWS_SUPPORT)
if (CreateThread(0,0,DistributePixelCacheClient,(void*) &client_socket,0,
&threadID) == (HANDLE) NULL)
ThrowFatalException(CacheFatalError,"UnableToCreateClientThread");
#else
Not implemented!
#endif
}
#else
magick_unreferenced(port);
magick_unreferenced(exception);
ThrowFatalException(MissingDelegateError,"DelegateLibrarySupportNotBuiltIn");
#endif
}
MagickPrivate int GetDistributeCacheFile(const DistributeCacheInfo *server_info)
{
assert(server_info != (DistributeCacheInfo *) NULL);
assert(server_info->signature == MagickSignature);
return(server_info->file);
}
MagickPrivate const char *GetDistributeCacheHostname(
const DistributeCacheInfo *server_info)
{
assert(server_info != (DistributeCacheInfo *) NULL);
assert(server_info->signature == MagickSignature);
return(server_info->hostname);
}
MagickPrivate int GetDistributeCachePort(const DistributeCacheInfo *server_info)
{
assert(server_info != (DistributeCacheInfo *) NULL);
assert(server_info->signature == MagickSignature);
return(server_info->port);
}
MagickPrivate MagickBooleanType OpenDistributePixelCache(
DistributeCacheInfo *server_info,Image *image)
{
MagickBooleanType
#ifdef __VMS
status=MagickTrue;
#else
status;
#endif
MagickOffsetType
count;
register unsigned char
*p;
unsigned char
message[MaxTextExtent];
assert(server_info != (DistributeCacheInfo *) NULL);
assert(server_info->signature == MagickSignature);
assert(image != (Image *) NULL);
assert(image->signature == MagickSignature);
p=message;
*p++='o';
(void) memcpy(p,&server_info->session_key,sizeof(server_info->session_key));
p+=sizeof(server_info->session_key);
(void) memcpy(p,&image->storage_class,sizeof(image->storage_class));
p+=sizeof(image->storage_class);
(void) memcpy(p,&image->colorspace,sizeof(image->colorspace));
p+=sizeof(image->colorspace);
(void) memcpy(p,&image->channels,sizeof(image->channels));
p+=sizeof(image->channels);
(void) memcpy(p,&image->columns,sizeof(image->columns));
p+=sizeof(image->columns);
(void) memcpy(p,&image->rows,sizeof(image->rows));
p+=sizeof(image->rows);
count=dpc_send(server_info->file,p-message,message);
if (count != (MagickOffsetType) (p-message))
return(MagickFalse);
status=MagickFalse;
count=dpc_read(server_info->file,sizeof(status),(unsigned char *) &status);
if (count != (MagickOffsetType) sizeof(status))
return(MagickFalse);
return(status);
}
MagickPrivate MagickOffsetType ReadDistributePixelCacheIndexes(
DistributeCacheInfo *server_info,const RectangleInfo *region,
const MagickSizeType length,unsigned char *indexes)
{
MagickOffsetType
count;
register unsigned char
*p;
unsigned char
message[MaxTextExtent];
assert(server_info != (DistributeCacheInfo *) NULL);
assert(server_info->signature == MagickSignature);
assert(region != (RectangleInfo *) NULL);
assert(indexes != (unsigned char *) NULL);
if (length > (MagickSizeType) SSIZE_MAX)
return(-1);
p=message;
*p++='R';
(void) memcpy(p,&server_info->session_key,sizeof(server_info->session_key));
p+=sizeof(server_info->session_key);
(void) memcpy(p,®ion->width,sizeof(region->width));
p+=sizeof(region->width);
(void) memcpy(p,®ion->height,sizeof(region->height));
p+=sizeof(region->height);
(void) memcpy(p,®ion->x,sizeof(region->x));
p+=sizeof(region->x);
(void) memcpy(p,®ion->y,sizeof(region->y));
p+=sizeof(region->y);
(void) memcpy(p,&length,sizeof(length));
p+=sizeof(length);
count=dpc_send(server_info->file,p-message,message);
if (count != (MagickOffsetType) (p-message))
return(-1);
return(dpc_read(server_info->file,length,indexes));
}
MagickPrivate MagickOffsetType ReadDistributePixelCachePixels(
DistributeCacheInfo *server_info,const RectangleInfo *region,
const MagickSizeType length,unsigned char *magick_restrict pixels)
{
MagickOffsetType
count;
register unsigned char
*p;
unsigned char
message[MaxTextExtent];
assert(server_info != (DistributeCacheInfo *) NULL);
assert(server_info->signature == MagickSignature);
assert(region != (RectangleInfo *) NULL);
assert(pixels != (unsigned char *) NULL);
if (length > (MagickSizeType) SSIZE_MAX)
return(-1);
p=message;
*p++='r';
(void) memcpy(p,&server_info->session_key,sizeof(server_info->session_key));
p+=sizeof(server_info->session_key);
(void) memcpy(p,®ion->width,sizeof(region->width));
p+=sizeof(region->width);
(void) memcpy(p,®ion->height,sizeof(region->height));
p+=sizeof(region->height);
(void) memcpy(p,®ion->x,sizeof(region->x));
p+=sizeof(region->x);
(void) memcpy(p,®ion->y,sizeof(region->y));
p+=sizeof(region->y);
(void) memcpy(p,&length,sizeof(length));
p+=sizeof(length);
count=dpc_send(server_info->file,p-message,message);
if (count != (MagickOffsetType) (p-message))
return(-1);
return(dpc_read(server_info->file,length,pixels));
}
MagickPrivate MagickBooleanType RelinquishDistributePixelCache(
DistributeCacheInfo *server_info)
{
MagickBooleanType
#ifdef __VMS
status = MagickTrue;
#else
status;
#endif
MagickOffsetType
count;
register unsigned char
*p;
unsigned char
message[MaxTextExtent];
assert(server_info != (DistributeCacheInfo *) NULL);
assert(server_info->signature == MagickSignature);
p=message;
*p++='d';
(void) memcpy(p,&server_info->session_key,sizeof(server_info->session_key));
p+=sizeof(server_info->session_key);
count=dpc_send(server_info->file,p-message,message);
if (count != (MagickOffsetType) (p-message))
return(MagickFalse);
count=dpc_read(server_info->file,sizeof(status),(unsigned char *) &status);
if (count != (MagickOffsetType) sizeof(status))
return(MagickFalse);
return(status);
}
MagickPrivate MagickOffsetType WriteDistributePixelCacheIndexes(
DistributeCacheInfo *server_info,const RectangleInfo *region,
const MagickSizeType length,const unsigned char *indexes)
{
MagickOffsetType
count;
register unsigned char
*p;
unsigned char
message[MaxTextExtent];
assert(server_info != (DistributeCacheInfo *) NULL);
assert(server_info->signature == MagickSignature);
assert(region != (RectangleInfo *) NULL);
assert(indexes != (unsigned char *) NULL);
if (length > (MagickSizeType) SSIZE_MAX)
return(-1);
p=message;
*p++='W';
(void) memcpy(p,&server_info->session_key,sizeof(server_info->session_key));
p+=sizeof(server_info->session_key);
(void) memcpy(p,®ion->width,sizeof(region->width));
p+=sizeof(region->width);
(void) memcpy(p,®ion->height,sizeof(region->height));
p+=sizeof(region->height);
(void) memcpy(p,®ion->x,sizeof(region->x));
p+=sizeof(region->x);
(void) memcpy(p,®ion->y,sizeof(region->y));
p+=sizeof(region->y);
(void) memcpy(p,&length,sizeof(length));
p+=sizeof(length);
count=dpc_send(server_info->file,p-message,message);
if (count != (MagickOffsetType) (p-message))
return(-1);
return(dpc_send(server_info->file,length,indexes));
}
MagickPrivate MagickOffsetType WriteDistributePixelCachePixels(
DistributeCacheInfo *server_info,const RectangleInfo *region,
const MagickSizeType length,const unsigned char *magick_restrict pixels)
{
MagickOffsetType
count;
register unsigned char
*p;
unsigned char
message[MaxTextExtent];
assert(server_info != (DistributeCacheInfo *) NULL);
assert(server_info->signature == MagickSignature);
assert(region != (RectangleInfo *) NULL);
assert(pixels != (const unsigned char *) NULL);
if (length > (MagickSizeType) SSIZE_MAX)
return(-1);
p=message;
*p++='w';
(void) memcpy(p,&server_info->session_key,sizeof(server_info->session_key));
p+=sizeof(server_info->session_key);
(void) memcpy(p,®ion->width,sizeof(region->width));
p+=sizeof(region->width);
(void) memcpy(p,®ion->height,sizeof(region->height));
p+=sizeof(region->height);
(void) memcpy(p,®ion->x,sizeof(region->x));
p+=sizeof(region->x);
(void) memcpy(p,®ion->y,sizeof(region->y));
p+=sizeof(region->y);
(void) memcpy(p,&length,sizeof(length));
p+=sizeof(length);
count=dpc_send(server_info->file,p-message,message);
if (count != (MagickOffsetType) (p-message))
return(-1);
return(dpc_send(server_info->file,length,pixels));
}