/* Copyright (C) 2001-2023 Artifex Software, Inc. All Rights Reserved. This software is provided AS-IS with no warranty, either express or implied. This software is distributed under license and may not be copied, modified or distributed except as expressly authorized under the terms of the license contained in the file LICENSE in this distribution. Refer to licensing information at http://www.artifex.com or contact Artifex Software, Inc., 39 Mesa Street, Suite 108A, San Francisco, CA 94129, USA, for further information. */ /* Dynamic Link Library interface for OS/2 and MS-Windows Ghostscript */ /* front end to gs.c */ /* This has been reimplemented to call the new DLL interface in iapi.h */ #ifdef _Windows #include #endif #ifdef __OS2__ #define INCL_DOS #define INCL_WIN #include #endif #include "stdpre.h" #include "iapi.h" /* Ghostscript interpreter public interface */ #include "string_.h" #include "ierrors.h" #include "gscdefs.h" #include "gstypes.h" #include "iref.h" #include "iminst.h" #include "imain.h" #include "gsdll.h" /* old DLL public interface */ /* MacGSView still requires that hwnd be exported through the old dll interface. We do that here, but expect to remove it when that client has been ported to the gsapi interface. */ #ifdef __MACOS__ extern HWND hwndtext; #endif /****** SINGLE-INSTANCE HACK ******/ /* GLOBAL WARNING */ GSDLL_CALLBACK pgsdll_callback = NULL; /* callback for messages and stdio to caller */ static gs_main_instance *pgs_minst = NULL; /****** SINGLE-INSTANCE HACK ******/ /* local functions */ static int GSDLLCALL gsdll_old_stdin(void *caller_handle, char *buf, int len); static int GSDLLCALL gsdll_old_stdout(void *caller_handle, const char *str, int len); static int GSDLLCALL gsdll_old_stderr(void *caller_handle, const char *str, int len); static int GSDLLCALL gsdll_old_poll(void *caller_handle); /* ---------- DLL exported functions ---------- */ /* arguments are: * 1. callback function for stdio and for notification of * sync_output, output_page and resize events * 2. window handle, used as parent. Use NULL if you have no window. * 3. argc * 4. argv */ int GSDLLEXPORT GSDLLAPI gsdll_init_with_encoding(GSDLL_CALLBACK callback, HWND hwnd, int argc, char * argv[], int encoding) { int code; if ((code = gsapi_new_instance(&pgs_minst, (void *)1)) < 0) return -1; gsapi_set_stdio(pgs_minst, gsdll_old_stdin, gsdll_old_stdout, gsdll_old_stderr); gsapi_set_poll(pgs_minst, gsdll_old_poll); /* ignore hwnd */ /* rest of MacGSView compatibilty hack */ #ifdef __MACOS__ hwndtext=hwnd; #endif /****** SINGLE-INSTANCE HACK ******/ pgsdll_callback = callback; /****** SINGLE-INSTANCE HACK ******/ code = gsapi_set_arg_encoding(pgs_minst, encoding); if (code >= 0) code = gsapi_init_with_args(pgs_minst, argc, argv); if (code == gs_error_Quit) { gsapi_exit(pgs_minst); return GSDLL_INIT_QUIT; } return code; } int GSDLLEXPORT GSDLLAPI gsdll_init(GSDLL_CALLBACK callback, HWND hwnd, int argc, char * argv[]) { return gsdll_init_with_encoding(callback, hwnd, argc, argv, GS_ARG_ENCODING_LOCAL); } #ifdef __WIN32__ int GSDLLEXPORT GSDLLAPI gsdll_initW(GSDLL_CALLBACK callback, HWND hwnd, int argc, wchar_t * argv[]) { return gsdll_init_with_encoding(callback, hwnd, argc, (char **)argv, GS_ARG_ENCODING_UTF16LE); } int GSDLLEXPORT GSDLLAPI gsdll_initA(GSDLL_CALLBACK callback, HWND hwnd, int argc, char * argv[]) { return gsdll_init_with_encoding(callback, hwnd, argc, argv, GS_ARG_ENCODING_LOCAL); } #endif /* if return value < 0, then error occured and caller should call */ /* gsdll_exit, then unload library */ int GSDLLEXPORT GSDLLAPI gsdll_execute_begin(void) { int exit_code; return gsapi_run_string_begin(pgs_minst, 0, &exit_code); } /* if return value < 0, then error occured and caller should call */ /* gsdll_execute_end, then gsdll_exit, then unload library */ int GSDLLEXPORT GSDLLAPI gsdll_execute_cont(const char * str, int len) { int exit_code; int code = gsapi_run_string_continue(pgs_minst, str, len, 0, &exit_code); if (code == gs_error_NeedInput) code = 0; /* this is not an error */ return code; } /* if return value < 0, then error occured and caller should call */ /* gsdll_exit, then unload library */ int GSDLLEXPORT GSDLLAPI gsdll_execute_end(void) { int exit_code; return gsapi_run_string_end(pgs_minst, 0, &exit_code); } int GSDLLEXPORT GSDLLAPI gsdll_exit(void) { int code = gsapi_exit(pgs_minst); gsapi_delete_instance(pgs_minst); return code; } /* Return revision numbers and strings of Ghostscript. */ /* Used for determining if wrong GSDLL loaded. */ /* This may be called before any other function. */ int GSDLLEXPORT GSDLLAPI gsdll_revision(const char ** product, const char ** copyright, long * revision, long * revisiondate) { if (product) *product = gs_product; if (copyright) *copyright = gs_copyright; if (revision) *revision = gs_revision; if (revisiondate) *revisiondate = gs_revisiondate; return 0; } static int GSDLLCALL gsdll_old_stdin(void *caller_handle, char *buf, int len) { return (*pgsdll_callback)(GSDLL_STDIN, buf, len); } static int GSDLLCALL gsdll_old_stdout(void *caller_handle, const char *str, int len) { return (*pgsdll_callback)(GSDLL_STDOUT, (char *)str, len); } static int GSDLLCALL gsdll_old_stderr(void *caller_handle, const char *str, int len) { return (*pgsdll_callback)(GSDLL_STDOUT, (char *)str, len); } static int GSDLLCALL gsdll_old_poll(void *caller_handle) { return (*pgsdll_callback)(GSDLL_POLL, NULL, 0); } /* end gsdll.c */