/* Copyright (C) 2004,2005,2006 (Nuno A. Fonseca) This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. Last rev: $Id: prologterms2c.c,v 1.4 2006-09-28 11:42:51 vsc Exp $ Comments: This file provides a set of functions to convert a prolog term to a C string and back. */ #define HAVE_STRING_H 1 #define HAVE_MALLOC_H 1 #define HAVE_UNISTD_H 1 #define HAVE_SYS_TIMES_H 1 #define HAVE_STDINT_H 1 #define HAVE_STDBOOL_H 1 #define HAVE_MPI_H 1 #include "prologterms2c.h" #include #include #if HAVE_STRING_H #include #endif #if HAVE_STDARG_H #include #endif #if HAVE_MALLOC_H #include #endif #if HAVE_MPI_H #ifdef COMPRESS #include "minilzo.h" #endif #ifndef Quote_illegal_f #define Quote_illegal_f 1 #define Ignore_ops_f 2 #define Handle_vars_f 4 #define Use_portray_f 8 #define To_heap_f 16 #endif #ifdef COMPRESS #endif size_t YAP_ExportTerm(YAP_Term inp, char * buf, size_t len); size_t YAP_ExportTerm(YAP_Term inp, char * buf, size_t len) { size_t len1; char * ext; if (!len) return 0; len1=len; ext=PL_record_external(inp,&len1); if (len1>len) { PL_erase_external(ext); return 0; } else { memcpy(buf,ext,len1); PL_erase_external(ext); return len1; } } YAP_Term YAP_ImportTerm(char * buf); YAP_Term YAP_ImportTerm(char * buf) { term_t term; term=PL_new_term_ref(); PL_recorded_external(buf,term); // PL_erase_external(buf); return term; } struct buffer_ds buffers[1024]; /*********************************************************************************************/ // prototypes void write_msg(const char *fun,const char *file, int line,const char *format, ...); size_t write_term_to_stream(const int fd,const YAP_Term term); YAP_Term read_term_from_stream(const int fd); /*********************************************************************************************/ /* * Writes a debug message containing the processid, function name, filename, line, and a user message */ void write_msg(const char *fun,const char *file, int line, const char *format, ...) { va_list ap; va_start(ap, format); /* Print the message to stderr */ fprintf(stderr, "[%d:%s in %s at line %d] ", getpid(),fun, file, line); vfprintf(stderr, format, ap); va_end(ap); } /********************************************************************************************* * Memory handling functions *********************************************************************************************/ /* * Adds 'space' to the size of the currently allocated buffer */ static void expand_buffer(const size_t space ) { BUFFER_PTR = realloc( BUFFER_PTR, BUFFER_SIZE + space ); if( BUFFER_PTR == NULL ) { YAP_Error(0,0,"Prolog2Term: Out of memory.\n"); #ifdef MPI MPI_Finalize(); #endif YAP_Exit( 1 ); } BUFFER_SIZE+=space; } /* * Changes the size of the buffer to contain at least newsize bytes */ void change_buffer_size(const size_t newsize) { if ( BUFFER_PTR == NULL ) { if ((BUFFER_PTR = malloc( BLOCK_SIZE < newsize ? newsize : BLOCK_SIZE)) == NULL) { YAP_Error(0,0,"Prolog2Term: Out of memory.\n"); #ifdef MPI MPI_Finalize(); #endif YAP_Exit( 1 ); } } else if ((BUFFER_SIZE>=BLOCK_SIZE && BUFFER_SIZE>=newsize) ) { return; } else if ((BUFFER_PTR = realloc( BUFFER_PTR, newsize)) == NULL) { YAP_Error(0,0,"Prolog2Term: Out of memory.\n"); #ifdef MPI MPI_Finalize(); #endif YAP_Exit( 1 ); } BUFFER_SIZE=newsize; } /********************************************************************************************* * I/O functions *********************************************************************************************/ /* * Function used by YAP to write a char to a string */ static void p2c_putt(const YAP_Term t) { // if( buffer.size==buffer.len+1 ) while ((BUFFER_LEN=YAP_ExportTerm(t, BUFFER_PTR, BUFFER_SIZE)) <= 0) { #ifdef DEBUG write_msg(__FUNCTION__,__FILE__,__LINE__,"p2c_putc:buffer expanded: size=%u pos=%u len=%u\n",BUFFER_SIZE,BUFFER_POS,BUFFER_LEN); #endif expand_buffer( BLOCK_SIZE ); } } /* * Function used by YAP to read a char from a string */ /* * Writes a term to a stream. */ size_t write_term_to_stream(const int fd,const YAP_Term term) { RESET_BUFFER(); printf("BUFFER_PTR=%p\n", BUFFER_PTR); p2c_putt(term); if (write(fd,(void*)BUFFER_PTR,BUFFER_LEN) < 0) { // write term YAP_Error(0,0,"Prolog2Term: IO error in write.\n"); return -1; } return BUFFER_LEN; } /* * Read a prolog term from a stream * (the prolog term must have been writen by the write_term_to_stream) */ YAP_Term read_term_from_stream(const int fd) { size_t size; RESET_BUFFER(); if (!read(fd,(void*)&size,sizeof(size_t))) { // read the size of the term YAP_Error(0,0,"Prolog2Term: IO error in read.\n"); } #ifdef DEBUG write_msg(__FUNCTION__,__FILE__,__LINE__,"read_term_from_stream>>>>size:%d\n",size); #endif if ( size> BUFFER_SIZE) expand_buffer(size-BUFFER_SIZE); if (!read(fd,BUFFER_PTR,size)) { YAP_Error(0,0,"Prolog2Term: IO error in read.\n"); }; // read term from stream return YAP_ImportTerm( BUFFER_PTR); } /********************************************************************************************* * Conversion: Prolog Term->char[] and char[]->Prolog Term *********************************************************************************************/ /* * Converts a term t into a string. * The ascii representation of t is * copied to ptr if it occupies less than size. */ char* term2string(char *const ptr, size_t *size, const YAP_Term t) { char *ret; RESET_BUFFER(); do { if (*size == 0) { *size = BUFFER_LEN = YAP_ExportTerm( t, BUFFER_PTR, BUFFER_SIZE );// canonical ret=BUFFER_PTR; if (BUFFER_LEN == 0) { expand_buffer(BLOCK_SIZE); } } else { *size = YAP_ExportTerm( t, ptr, BUFFER_SIZE );// canonical ret=ptr; } } while (*size <= 0); return ret; } /* * Converts a string with a ascci representation of a term into a Prolog term. */ YAP_Term string2term(char *const ptr,const size_t *size) { YAP_Term t; t = YAP_ImportTerm( ptr ); if ( t==FALSE ) { write_msg(__FUNCTION__,__FILE__,__LINE__,"FAILED string2term>>>>size:%lx %d\n",t,*size); exit(1); } return t; } #endif /* HAVE_MPI_H */