From 464d548d7173d964b028941400b7e6a740fb7f94 Mon Sep 17 00:00:00 2001 From: Erez Geva Date: Fri, 6 May 2022 20:36:57 +0200 Subject: [PATCH] Add argcargv.i to more languages: Perl 5, Tcl, PHP Signed-off-by: Erez Geva --- Lib/perl5/argcargv.i | 31 +++++++++++++++++++++++++++++++ Lib/php/argcargv.i | 40 ++++++++++++++++++++++++++++++++++++++++ Lib/tcl/argcargv.i | 28 ++++++++++++++++++++++++++++ 3 files changed, 99 insertions(+) create mode 100644 Lib/perl5/argcargv.i create mode 100644 Lib/php/argcargv.i create mode 100644 Lib/tcl/argcargv.i diff --git a/Lib/perl5/argcargv.i b/Lib/perl5/argcargv.i new file mode 100644 index 000000000..de7b626e0 --- /dev/null +++ b/Lib/perl5/argcargv.i @@ -0,0 +1,31 @@ +/* ------------------------------------------------------------ + * --- Argc & Argv --- + * ------------------------------------------------------------ */ + +%typemap(default) (int ARGC, char **ARGV) { + $1 = 0; $2 = NULL; +} + +%typemap(in) (int ARGC, char **ARGV) { + int i; + I32 len; + AV *av = (AV *)SvRV($input); + if (SvTYPE(av) != SVt_PVAV) { + SWIG_croak("in method '$symname', Expecting reference to argv array"); + goto fail; + } + len = av_len(av) + 1; + $1 = ($1_ltype) len; + $2 = (char **) malloc((len+1)*sizeof(char *)); + for (i = 0; i < len; i++) { + SV **tv = av_fetch(av, i, 0); + $2[i] = SvPV_nolen(*tv); + } + $2[i] = NULL; +} + +%typemap(freearg) (int ARGC, char **ARGV) { + if ($2 != NULL) { + free((void *)$2); + } +} diff --git a/Lib/php/argcargv.i b/Lib/php/argcargv.i new file mode 100644 index 000000000..14b0cf2c9 --- /dev/null +++ b/Lib/php/argcargv.i @@ -0,0 +1,40 @@ +/* ------------------------------------------------------------ + * --- Argc & Argv --- + * ------------------------------------------------------------ */ + +%typemap(default) (int ARGC, char **ARGV) { + $1 = 0; $2 = NULL; +} + +%typemap(in) (int ARGC, char **ARGV) { + int len, i; + zval *val; + zend_array *ar; + if (Z_TYPE($input) != IS_ARRAY) { + SWIG_PHP_Error(E_ERROR, "Type error in '$symname'. Expected array"); + goto fail; + } + ar = Z_ARR($input); + len = zend_array_count(ar); + $1 = ($1_ltype) len; + $2 = (char **) malloc((len+1)*sizeof(char *)); + i = 0; + ZEND_HASH_FOREACH_VAL(ar, val) { + if (Z_TYPE(*val) != IS_STRING) { + SWIG_PHP_Error(E_ERROR, "Array must use strings only, in '$symname'."); + goto fail; + } + if (i == len) { + SWIG_PHP_Error(E_ERROR, "Array is bigger than zend report in '$symname'."); + goto fail; + } + $2[i++] = Z_STRVAL(*val); + } ZEND_HASH_FOREACH_END(); + $2[i] = NULL; +} + +%typemap(freearg) (int ARGC, char **ARGV) { + if ($2 != NULL) { + free((void *)$2); + } +} diff --git a/Lib/tcl/argcargv.i b/Lib/tcl/argcargv.i new file mode 100644 index 000000000..e43f2e657 --- /dev/null +++ b/Lib/tcl/argcargv.i @@ -0,0 +1,28 @@ +/* ------------------------------------------------------------ + * --- Argc & Argv --- + * ------------------------------------------------------------ */ + +%typemap(default) (int ARGC, char **ARGV) { + $1 = 0; $2 = NULL; +} + +%typemap(in) (int ARGC, char **ARGV) { + int i, nitems; + Tcl_Obj **listobjv; + if (Tcl_ListObjGetElements(interp, $input, &nitems, &listobjv) == TCL_ERROR) { + SWIG_exception_fail(SWIG_ValueError, "in method '$symname', Expecting list of argv"); + goto fail; + } + $1 = ($1_ltype) nitems; + $2 = (char **) malloc((nitems+1)*sizeof(char *)); + for (i = 0; i < nitems; i++) { + $2[i] = Tcl_GetStringFromObj(listobjv[i], NULL); + } + $2[i] = NULL; +} + +%typemap(freearg) (int ARGC, char **ARGV) { + if ($2 != NULL) { + free((void *)$2); + } +}