mirror of
https://git.savannah.gnu.org/git/guile.git
synced 2025-06-13 15:10:34 +02:00
Add `scm_c_value_ref' to allow access to multiple returned values from C
Based on a patch by Julian Graham <julian@member.fsf.org> * libguile/values.c, libguile/values.h (scm_c_value_ref): New function. * doc/ref/api-control.texi (Multiple Values): Add documentation. * test-suite/standalone/test-scm-values.c: New test program. * test-suite/standalone/Makefile.am: Add test-scm-values test.
This commit is contained in:
parent
f0007cade0
commit
1ceeca0a76
5 changed files with 109 additions and 0 deletions
|
@ -785,6 +785,13 @@ the current implementation that object shares structure with
|
||||||
@var{args}, so @var{args} should not be modified subsequently.
|
@var{args}, so @var{args} should not be modified subsequently.
|
||||||
@end deffn
|
@end deffn
|
||||||
|
|
||||||
|
@deffn {C Function} scm_c_value_ref (values, idx)
|
||||||
|
Returns the value at the position specified by @var{idx} in
|
||||||
|
@var{values}. Note that @var{values} will ordinarily be a
|
||||||
|
multiple-values object, but it need not be. Any other object
|
||||||
|
represents a single value (itself), and is handled appropriately.
|
||||||
|
@end deffn
|
||||||
|
|
||||||
@rnindex call-with-values
|
@rnindex call-with-values
|
||||||
@deffn {Scheme Procedure} call-with-values producer consumer
|
@deffn {Scheme Procedure} call-with-values producer consumer
|
||||||
Calls its @var{producer} argument with no values and a
|
Calls its @var{producer} argument with no values and a
|
||||||
|
|
|
@ -67,6 +67,31 @@ print_values (SCM obj, SCM pwps)
|
||||||
return SCM_UNSPECIFIED;
|
return SCM_UNSPECIFIED;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
SCM
|
||||||
|
scm_c_value_ref (SCM obj, size_t idx)
|
||||||
|
{
|
||||||
|
if (SCM_LIKELY (SCM_VALUESP (obj)))
|
||||||
|
{
|
||||||
|
SCM values = scm_struct_ref (obj, SCM_INUM0);
|
||||||
|
size_t i = idx;
|
||||||
|
while (SCM_LIKELY (scm_is_pair (values)))
|
||||||
|
{
|
||||||
|
if (i == 0)
|
||||||
|
return SCM_CAR (values);
|
||||||
|
values = SCM_CDR (values);
|
||||||
|
i--;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (idx == 0)
|
||||||
|
return obj;
|
||||||
|
|
||||||
|
scm_error (scm_out_of_range_key,
|
||||||
|
"scm_c_value_ref",
|
||||||
|
"Too few values in ~S to access index ~S",
|
||||||
|
scm_list_2 (obj, scm_from_unsigned_integer (idx)),
|
||||||
|
scm_list_1 (scm_from_unsigned_integer (idx)));
|
||||||
|
}
|
||||||
|
|
||||||
SCM_DEFINE (scm_values, "values", 0, 0, 1,
|
SCM_DEFINE (scm_values, "values", 0, 0, 1,
|
||||||
(SCM args),
|
(SCM args),
|
||||||
"Delivers all of its arguments to its continuation. Except for\n"
|
"Delivers all of its arguments to its continuation. Except for\n"
|
||||||
|
|
|
@ -33,6 +33,7 @@ SCM_API SCM scm_values_vtable;
|
||||||
SCM_INTERNAL void scm_i_extract_values_2 (SCM obj, SCM *p1, SCM *p2);
|
SCM_INTERNAL void scm_i_extract_values_2 (SCM obj, SCM *p1, SCM *p2);
|
||||||
|
|
||||||
SCM_API SCM scm_values (SCM args);
|
SCM_API SCM scm_values (SCM args);
|
||||||
|
SCM_API SCM scm_c_value_ref (SCM values, size_t idx);
|
||||||
SCM_INTERNAL void scm_init_values (void);
|
SCM_INTERNAL void scm_init_values (void);
|
||||||
|
|
||||||
#endif /* SCM_VALUES_H */
|
#endif /* SCM_VALUES_H */
|
||||||
|
|
|
@ -189,6 +189,13 @@ test_scm_to_latin1_string_LDADD = $(LIBGUILE_LDADD)
|
||||||
check_PROGRAMS += test-scm-to-latin1-string
|
check_PROGRAMS += test-scm-to-latin1-string
|
||||||
TESTS += test-scm-to-latin1-string
|
TESTS += test-scm-to-latin1-string
|
||||||
|
|
||||||
|
# test-scm-values
|
||||||
|
test_scm_values_SOURCES = test-scm-values.c
|
||||||
|
test_scm_values_CFLAGS = ${test_cflags}
|
||||||
|
test_scm_values_LDADD = $(LIBGUILE_LDADD)
|
||||||
|
check_PROGRAMS += test-scm-values
|
||||||
|
TESTS += test-scm-values
|
||||||
|
|
||||||
if HAVE_SHARED_LIBRARIES
|
if HAVE_SHARED_LIBRARIES
|
||||||
|
|
||||||
# test-extensions
|
# test-extensions
|
||||||
|
|
69
test-suite/standalone/test-scm-values.c
Normal file
69
test-suite/standalone/test-scm-values.c
Normal file
|
@ -0,0 +1,69 @@
|
||||||
|
/* Copyright (C) 2012 Free Software Foundation, Inc.
|
||||||
|
*
|
||||||
|
* This library is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU Lesser General Public License
|
||||||
|
* as published by the Free Software Foundation; either version 3 of
|
||||||
|
* the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This library 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
|
||||||
|
* Lesser General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Lesser General Public
|
||||||
|
* License along with this library; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
|
||||||
|
* 02110-1301 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifdef HAVE_CONFIG_H
|
||||||
|
# include <config.h>
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#include <assert.h>
|
||||||
|
#include <libguile.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
|
||||||
|
static void
|
||||||
|
test_scm_c_value_ref_on_multiple_values ()
|
||||||
|
{
|
||||||
|
SCM values = scm_values (scm_list_3 (scm_from_latin1_string ("foo"),
|
||||||
|
scm_from_latin1_string ("bar"),
|
||||||
|
scm_from_latin1_string ("baz")));
|
||||||
|
|
||||||
|
char *foo = scm_to_latin1_string (scm_c_value_ref (values, 0));
|
||||||
|
char *bar = scm_to_latin1_string (scm_c_value_ref (values, 1));
|
||||||
|
char *baz = scm_to_latin1_string (scm_c_value_ref (values, 2));
|
||||||
|
|
||||||
|
assert (strcmp (foo, "foo") == 0);
|
||||||
|
assert (strcmp (bar, "bar") == 0);
|
||||||
|
assert (strcmp (baz, "baz") == 0);
|
||||||
|
|
||||||
|
free (foo);
|
||||||
|
free (bar);
|
||||||
|
free (baz);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
test_scm_c_value_ref_on_a_single_value ()
|
||||||
|
{
|
||||||
|
SCM value = scm_from_latin1_string ("foo");
|
||||||
|
char *foo = scm_to_latin1_string (scm_c_value_ref (value, 0));
|
||||||
|
assert (strcmp (foo, "foo") == 0);
|
||||||
|
free (foo);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
tests (void *data, int argc, char **argv)
|
||||||
|
{
|
||||||
|
test_scm_c_value_ref_on_multiple_values ();
|
||||||
|
test_scm_c_value_ref_on_a_single_value ();
|
||||||
|
}
|
||||||
|
|
||||||
|
int
|
||||||
|
main (int argc, char *argv[])
|
||||||
|
{
|
||||||
|
scm_boot_guile (argc, argv, tests, NULL);
|
||||||
|
return 0;
|
||||||
|
}
|
Loading…
Add table
Add a link
Reference in a new issue