summaryrefslogtreecommitdiff
path: root/gpdl
diff options
context:
space:
mode:
authorHenry Stiles <henry.stiles@artifex.com>2017-05-17 15:27:23 -0600
committerHenry Stiles <henry.stiles@artifex.com>2017-05-19 12:52:20 -0600
commitd7cade08ad36382a0ed5dfa72fcb9a0e9a822417 (patch)
tree8ea128943b9ea322682d9363e2f31a21381622bf /gpdl
parent00d5670774e9c56cb8bf6e9f877756c82d10536e (diff)
downloadghostpdl-d7cade08ad36382a0ed5dfa72fcb9a0e9a822417.tar.gz
Remove shared device psi implementation and improve documentation.
The experimental gpdl build now only supports running PostScript and PDF in a seperate context with it's own device using the normal API. At this time it only runs files detected as PostScript or PDF in the default device with no arguments. The previous setup did not work properly and if we are to have a shared device it should be implemented anew anyway. Also improve the documentation for how languages plug into the language switching system.
Diffstat (limited to 'gpdl')
-rw-r--r--gpdl/psi/psitop.c498
1 files changed, 54 insertions, 444 deletions
diff --git a/gpdl/psi/psitop.c b/gpdl/psi/psitop.c
index af3ef0dfc..356292d67 100644
--- a/gpdl/psi/psitop.c
+++ b/gpdl/psi/psitop.c
@@ -55,22 +55,11 @@ extern int zflush(i_ctx_t *);
* PS interpreter instance: derived from pl_interp_implementation_t
*/
typedef struct ps_interp_instance_s {
- pl_interp_implementation_t pl; /* common part: must be first */
- gs_memory_t *plmemory; /* memory allocator to use with pl objects */
- gs_main_instance *minst; /* PS interp main instance */
- pl_page_action_t pre_page_action; /* action before page out */
- void *pre_page_closure; /* closure to call pre_page_action with */
- pl_page_action_t post_page_action; /* action before page out */
- void *post_page_closure; /* closure to call post_page_action with */
- bool fresh_job; /* true if we are starting a new job */
- bool pdf_stream; /* current stream is pdf */
- char pdf_file_name[gp_file_name_sizeof];
- FILE * pdf_filep; /* temporary file for writing out pdf file */
- ref job_save;
+ pl_interp_implementation_t pl;
} ps_interp_instance_t;
/* Get implemtation's characteristics */
-const pl_interp_characteristics_t * /* always returns a descriptor */
+static const pl_interp_characteristics_t * /* always returns a descriptor */
ps_impl_characteristics(
const pl_interp_implementation_t *impl /* implementation of interpereter to alloc */
)
@@ -93,492 +82,113 @@ ps_impl_characteristics(
return &ps_characteristics;
}
-/* Don't need to do anything to PS interpreter */
-static int /* ret 0 ok, else -ve error code */
-ps_impl_allocate_interp(
- pl_interp_t **interp, /* RETURNS abstract interpreter struct */
- const pl_interp_implementation_t *impl, /* implementation of interpereter to alloc */
- gs_memory_t *mem /* allocator to allocate interp from */
-)
-{
-
- static pl_interp_t pi; /* there's only one interpreter */
- /* There's only one PS interp, so return the static */
- *interp = &pi;
- return 0; /* success */
-}
-
-/* Do per-instance interpreter allocation/init. No device is set yet */
-static int /* ret 0 ok, else -ve error code */
-ps_impl_allocate_interp_instance(
- pl_interp_implementation_t **instance, /* RETURNS instance struct */
- pl_interp_t *interp, /* dummy interpreter */
- gs_memory_t *mem /* allocator to allocate instance from */
-)
-{
- int code = 0, exit_code;
- const char *argv[] = {
- "",
- "-dNOPAUSE",
-#ifndef DEBUG
- "-dQUIET",
-#else
- "-dOSTACKPRINT", // NB: debuggging postscript Needs to be removed.
- "-dESTACKPRINT", // NB: debuggging postscript Needs to be removed.
-#endif
- };
-
- int argc = countof(argv);
-
- ps_interp_instance_t *psi /****** SHOULD HAVE A STRUCT DESCRIPTOR ******/
- = (ps_interp_instance_t *)
- gs_alloc_bytes( mem,
- sizeof(ps_interp_instance_t),
- "ps_allocate_interp_instance(ps_interp_instance_t)"
- );
-
- /* If allocation error, deallocate & return */
- if (!psi) {
- return gs_error_VMerror;
- }
- /* Initialize for pl_main_universe_dnit/pl_deallocate_interp_instance
- in case of gs_main_init_with_args returns with error code. */
- psi->pl.interp = interp;
- /* Setup pointer to mem used by PostScript */
- psi->plmemory = mem;
- psi->minst = gs_main_alloc_instance(mem->non_gc_memory);
-
- *instance = (pl_interp_implementation_t *)psi;
- code = gs_main_init_with_args(psi->minst, argc, (char**)argv);
- if (code<0)
- return code;
-
- /* General init of PS interp instance */
-
- if ((code = gs_main_run_string_begin(psi->minst, 0, &exit_code, &psi->minst->error_object)) < 0)
- return exit_code;
-
- {
- gs_gstate *pgs = psi->minst->i_ctx_p->pgs;
- gsicc_init_iccmanager(pgs);
- }
- /* inialize fresh job to false so that we can check for a pdf
- file next job. */
- psi->fresh_job = true;
- /* default is a postscript stream */
- psi->pdf_stream = false;
-
- /* Return success */
- return 0;
-}
-
-/* NB this pointer should be placed in the ps instance */
-
-/* Set a client language into an interpreter instance */
-static int /* ret 0 ok, else -ve error code */
-ps_impl_set_client_instance(
- pl_interp_implementation_t *instance, /* interp instance to use */
- pl_interp_implementation_t *client, /* client to set */
- pl_interp_instance_clients_t which_client
-)
-{
- return 0;
-}
-
-/* Set an interpreter instance's pre-page action */
-static int /* ret 0 ok, else -ve err */
-ps_impl_set_pre_page_action(
- pl_interp_implementation_t *instance, /* interp instance to use */
- pl_page_action_t action, /* action to execute (rets 1 to abort w/o err) */
- void *closure /* closure to call action with */
-)
-{
- ps_interp_instance_t *psi = (ps_interp_instance_t *)instance;
- psi->pre_page_action = action;
- psi->pre_page_closure = closure;
- return 0;
-}
-
-/* Set an interpreter instance's post-page action */
-static int /* ret 0 ok, else -ve err */
-ps_impl_set_post_page_action(
- pl_interp_implementation_t *instance, /* interp instance to use */
- pl_page_action_t action, /* action to execute */
- void *closure /* closure to call action with */
-)
-{
- ps_interp_instance_t *psi = (ps_interp_instance_t *)instance;
- psi->post_page_action = action;
- psi->post_page_closure = closure;
- return 0;
+/* Do per-instance interpreter allocation/init. */
+static int
+ps_impl_allocate_interp_instance(pl_interp_implementation_t *impl, gs_memory_t *mem)
+{
+ ps_interp_instance_t *psi
+ = (ps_interp_instance_t *)gs_alloc_bytes( mem,
+ sizeof(ps_interp_instance_t),
+ "ps_impl_allocate_interp_instance");
+
+ int code;
+
+ if (!psi)
+ return gs_error_VMerror;
+
+ code = gsapi_new_instance(&impl->interp_client_data, NULL);
+ if (code < 0)
+ gs_free_object(mem, psi, "ps_impl_allocate_interp_instance");
+ return code;
}
/* Set a device into an interpreter instance */
-static int /* ret 0 ok, else -ve error code */
-ps_impl_set_device(
- pl_interp_implementation_t *instance, /* interp instance to use */
- gx_device *device /* device to set (open or closed) */
-)
+static int
+ps_impl_set_device(pl_interp_implementation_t *impl, gx_device *device)
{
- int code = 0;
- int exit_code = 0;
- ps_interp_instance_t *psi = (ps_interp_instance_t *)instance;
- gs_gstate *pgs = psi->minst->i_ctx_p->pgs;
-
- /* Initialize device ICC profile */
- code = gsicc_init_device_profile_struct(device, NULL, 0);
- if (code < 0)
- return code;
-
- /* Set the device into the gstate */
- code = gs_setdevice_no_erase(pgs, device);
- if (code < 0)
- return code;
-
- /* install a screen appropriate for the device */
- {
- const char *screen_str = ".setdefaultscreen\n";
- code = gsapi_run_string_continue(psi->plmemory->gs_lib_ctx,
- screen_str, strlen(screen_str),
- 0, &exit_code);
- /* needs more input this is not an error */
- if ( code == gs_error_NeedInput )
- code = 0;
-
- if (code < 0)
- return code;
- }
-
- /* this shouldn't fail */
- pgs = psi->minst->i_ctx_p->pgs; /* stuff moves if a GC happens during run_string */
- code = gs_erasepage(pgs);
- if (code < 0)
- return code;
-
- return exit_code;
+ /* Nothing to PS/PDF manages it's own device */
+ return 0;
}
-gs_main_instance *ps_impl_get_minst( const gs_memory_t *mem )
-{
- ps_interp_instance_t *psi = (ps_interp_instance_t *)get_interpreter_from_memory(mem);
- return psi->minst;
-}
/* Prepare interp instance for the next "job" */
-static int /* ret 0 ok, else -ve error code */
-ps_impl_init_job(
- pl_interp_implementation_t *instance /* interp instance to start job in */
-)
+static int
+ps_impl_init_job(pl_interp_implementation_t *impl)
{
- ps_interp_instance_t *psi = (ps_interp_instance_t *)instance;
- static const char *buf = "\004"; /* use ^D to start a new encapsulated job */
- int exit_code;
-
- /* starting a new job */
- psi->fresh_job = true;
- gsapi_run_string_continue(psi->plmemory->gs_lib_ctx, buf, strlen(buf), 0, &exit_code); /* ^D */
- return 0;
+ return gsapi_init_with_args(impl->interp_client_data, 0, NULL);
}
-/* Parse a buffer full of data */
-static int /* ret 0 or +ve if ok, else -ve error code */
-ps_impl_process(
- pl_interp_implementation_t *instance, /* interp instance to process data job in */
- stream_cursor_read *cursor /* data to process */
-)
+/* Not complete. */
+static int
+ps_impl_process_file(pl_interp_implementation_t *impl, char *filename)
{
- ps_interp_instance_t *psi = (ps_interp_instance_t *)instance;
- int code, exit_code;
- uint avail = cursor->limit - cursor->ptr;
- /* if we are at the beginning of a job check for pdf and set
- appropriate state variables to process either a pdf or ps
- job */
- if ( psi->fresh_job ) {
- const char pdf_idstr[] = "%PDF-1.";
- /* do we have enough data? */
- const uint pdf_idstr_len = strlen(pdf_idstr);
- if ( avail < pdf_idstr_len )
- /* more data. NB update ptr ?? */
- return 0;
- else
- /* compare beginning of stream with pdf id */
- if ( !strncmp(pdf_idstr, (const char *)cursor->ptr + 1, pdf_idstr_len) ) {
- char fmode[4];
- /* open the temporary pdf file. If the file open
- fails PDF fails and we allow the job to be sent
- to postscript and generate an error. It turns
- out this is easier than restoring the state and
- returning */
- strcpy(fmode, "w+");
- strcat(fmode, gp_fmode_binary_suffix);
- psi->pdf_filep = gp_open_scratch_file(psi->plmemory,
- gp_scratch_file_name_prefix,
- psi->pdf_file_name,
- fmode);
- if ( psi->pdf_filep == NULL )
- psi->pdf_stream = false;
- else
- psi->pdf_stream = true;
- }
- else
- psi->pdf_stream = false;
- /* we only check for pdf at the beginning of the job */
- psi->fresh_job = false;
- }
-
- /* for a pdf stream we append to the open pdf file but for
- postscript we hand it directly to the ps interpreter. PDF
- files are processed subsequently, at end job time */
- code = 0;
- if ( psi->pdf_stream ) {
- uint bytes_written = fwrite((cursor->ptr + 1), 1, avail, psi->pdf_filep);
- if ( bytes_written != avail )
- code = gs_error_invalidfileaccess;
- } else {
- /* Send the buffer to Ghostscript */
- code = gsapi_run_string_continue(psi->plmemory->gs_lib_ctx, (const char *)(cursor->ptr + 1),
- avail, 0, &exit_code);
- /* needs more input this is not an error */
- if ( code == gs_error_NeedInput )
- code = 0;
- /* error - I guess it gets "exit code" - nonsense */
- if ( code < 0 )
- code = exit_code;
- }
- /* update the cursor */
- cursor->ptr += avail;
- /* flush stdout on error. */
- if (code < 0)
- zflush(psi->minst->i_ctx_p);
- /* return the exit code */
- return code;
+ /* NB incomplete */
+ int exit_code;
+ return gsapi_run_file(impl->interp_client_data, filename, 0, &exit_code);
}
-/* Skip to end of job ret 1 if done, 0 ok but EOJ not found, else -ve error code */
+/* Not implemented */
static int
-ps_impl_flush_to_eoj(
- pl_interp_implementation_t *instance, /* interp instance to flush for */
- stream_cursor_read *cursor /* data to process */
-)
+ps_impl_flush_to_eoj(pl_interp_implementation_t *impl, stream_cursor_read *cursor)
{
- const byte *p = cursor->ptr;
- const byte *rlimit = cursor->limit;
-
- /* Skip to, but leave UEL in buffer for PJL to find later */
- for (; p < rlimit; ++p)
- if (p[1] == '\033') {
- uint avail = rlimit - p;
-
- if (memcmp(p + 1, "\033%-12345X", min(avail, 9)))
- continue;
- if (avail < 9)
- break;
- cursor->ptr = p;
- return 1; /* found eoj */
- }
- cursor->ptr = p;
- return 0; /* need more */
+ return 0;
}
/* Parser action for end-of-file */
-static int /* ret 0 or +ve if ok, else -ve error code */
-ps_impl_process_eof(
- pl_interp_implementation_t *instance /* interp instance to process data job in */
-)
+static int
+ps_impl_process_eof(pl_interp_implementation_t *impl)
{
- int code = 0;
-
- return code;
+ return 0;
}
/* Report any errors after running a job */
-static int /* ret 0 ok, else -ve error code */
-ps_impl_report_errors(pl_interp_implementation_t *instance, /* interp instance to wrap up job in */
+static int
+ps_impl_report_errors(pl_interp_implementation_t *impl, /* interp instance to wrap up job in */
int code, /* prev termination status */
long file_position, /* file position of error, -1 if unknown */
bool force_to_cout /* force errors to cout */
)
{
- /* ps_interp_instance_t *psi = (ps_interp_instance_t *)instance;
- */
- return code;
+ return 0;
}
/* Wrap up interp instance after a "job" */
-static int /* ret 0 ok, else -ve error code */
-ps_impl_dnit_job(
- pl_interp_implementation_t *instance /* interp instance to wrap up job in */
-)
+static int
+ps_impl_dnit_job(pl_interp_implementation_t *impl)
{
- int code = 0;
- int exit_code = 0;
- static const char *buf = "\n.endjob\n"; /* restore to initial state, non-encapsualted */
- ps_interp_instance_t *psi = (ps_interp_instance_t *)instance;
-
- /* take care of a stored pdf file */
- if ( psi->pdf_stream ) {
-
- /*
- * Hex encode to avoid problems with window's directory
- * separators '\' being interpreted in postscript as escape
- * sequences. The buffer length is the maximum file name size
- * (gp_file_name_sizeof) * 2 (hex encoding) + 2 (hex
- * delimiters '<' and '>') + the rest of the run command 7
- * (space + (run) + new line + null).
- */
- char buf[gp_file_name_sizeof * 2 + 2 + 7];
- const char *run_str = " run\n";
- const char *hex_digits = "0123456789ABCDEF";
- char *pd = buf; /* destination */
- const char *ps = psi->pdf_file_name; /* source */
-
- *pd = '<'; pd++;
- while (*ps) {
- *pd = hex_digits[*ps >> 4 ]; pd++;
- *pd = hex_digits[*ps & 0xf]; pd++;
- ps++;
- }
- *pd = '>'; pd++;
- strcpy(pd, run_str);
-
- /* at this point we have finished writing the spooled pdf file
- and we need to close it */
- fclose(psi->pdf_filep);
-
- /* Send the buffer to Ghostscript */
- code = gsapi_run_string_continue(psi->plmemory->gs_lib_ctx, buf, strlen(buf), 0, &exit_code);
-
- /* indicate we are done with the pdf stream */
- psi->pdf_stream = false;
- unlink(psi->pdf_file_name);
- /* handle errors... normally job deinit failures are
- considered fatal but pdf runs the spooled job when the job
- is deinitialized so handle error processing here and return code is always 0. */
- if (( code < 0) && (code != gs_error_NeedInput)) {
- errprintf(psi->plmemory, "PDF interpreter exited with exit code %d\n", exit_code);
- errprintf(psi->plmemory, "Flushing to EOJ\n");
- }
- code = 0;
- }
-
- /* We use string_end to send an EOF in case the job was reading in a loop */
- gsapi_run_string_end(psi->plmemory->gs_lib_ctx, 0, &exit_code); /* sends EOF to PS process */
- gsapi_run_string_begin(psi->plmemory->gs_lib_ctx, 0, &exit_code); /* prepare to send .endjob */
- gsapi_run_string_continue(psi->plmemory->gs_lib_ctx, buf, strlen(buf), 0, &exit_code); /* .endjob */
- /* Note the above will restore to the server save level and will not be encapsulated */
-
- /* Flush stdout. */
- zflush(psi->minst->i_ctx_p);
-
return 0;
}
/* Remove a device from an interpreter instance */
-static int /* ret 0 ok, else -ve error code */
-ps_impl_remove_device(
- pl_interp_implementation_t *instance /* interp instance to use */
-)
+static int
+ps_impl_remove_device(pl_interp_implementation_t *impl)
{
- /* Assuming the interpreter's stack contains a single graphic state.
- Otherwise this procedure is not effective.
- The Postscript job server logic must provide that.
- */
- ps_interp_instance_t *psi = (ps_interp_instance_t *)instance;
- gs_gstate *pgs = psi->minst->i_ctx_p->pgs;
- int code = gs_nulldevice(pgs);
-
- if ( code < 0 )
- dprintf1("error code %d installing nulldevice, continuing\n", code );
return 0;
}
/* Deallocate a interpreter instance */
-static int /* ret 0 ok, else -ve error code */
-ps_impl_deallocate_interp_instance(
- pl_interp_implementation_t *instance /* instance to dealloc */
-)
-{
- int code = 0, exit_code;
- ps_interp_instance_t *psi = (ps_interp_instance_t *)instance;
- gs_memory_t *mem = psi->plmemory;
-
- /* do total dnit of interp state */
- code = gsapi_run_string_end(mem->gs_lib_ctx, 0, &exit_code);
-
- gsapi_exit(psi->minst);
-
- gs_free_object(mem, psi, "ps_impl_deallocate_interp_instance(ps_interp_instance_t)");
-
- return (code < 0) ? exit_code : 0;
-}
-
-/* Do static deinit of PS interpreter */
-static int /* ret 0 ok, else -ve error code */
-ps_impl_deallocate_interp(
- pl_interp_t *interp /* interpreter to deallocate */
-)
-{
- /* nothing to do */
- return 0;
-}
-
-/*
- * End-of-page called back by PS
- */
-int
-ps_end_page_top(const gs_memory_t *mem, int num_copies, bool flush)
+static int
+ps_impl_deallocate_interp_instance(pl_interp_implementation_t *impl)
{
- pl_interp_implementation_t *instance = get_interpreter_from_memory(mem);
- ps_interp_instance_t *psi = (ps_interp_instance_t *)instance;
- int code = 0;
-
- if (psi == 0)
- return 0;
-
- /* do pre-page action */
- if (psi->pre_page_action) {
- code = psi->pre_page_action(instance, psi->pre_page_closure);
- if (code < 0)
- return code;
- if (code != 0)
- return 0; /* code > 0 means abort w/no error */
- }
- /* output the page */
- code = gs_output_page(psi->minst->i_ctx_p->pgs, num_copies, flush);
- if (code < 0)
- return code;
-
- /* Flush stdout. */
- zflush(psi->minst->i_ctx_p);
-
- /* do post-page action */
- if (psi->post_page_action) {
- code = psi->post_page_action(instance, psi->post_page_closure);
- if (code < 0)
- return code;
- }
-
- return 0;
+ int code = gsapi_exit(impl->interp_client_data);
+ gsapi_delete_instance(impl->interp_client_data);
+ return code;
}
/* Parser implementation descriptor */
const pl_interp_implementation_t ps_implementation = {
ps_impl_characteristics,
- ps_impl_allocate_interp,
ps_impl_allocate_interp_instance,
- ps_impl_set_client_instance,
- ps_impl_set_pre_page_action,
- ps_impl_set_post_page_action,
ps_impl_set_device,
ps_impl_init_job,
- NULL, /* process_file */
- ps_impl_process,
+ ps_impl_process_file,
+ NULL, /* process */
ps_impl_flush_to_eoj,
ps_impl_process_eof,
ps_impl_report_errors,
ps_impl_dnit_job,
ps_impl_remove_device,
ps_impl_deallocate_interp_instance,
- ps_impl_deallocate_interp
+ NULL
};