mirror of
git://git.sv.gnu.org/emacs.git
synced 2025-12-26 23:31:55 -08:00
Most of this change is to boilerplate commentary such as license URLs. This change was prompted by ftp://ftp.gnu.org's going-away party, planned for November. Change these FTP URLs to https://ftp.gnu.org instead. Make similar changes for URLs to other organizations moving away from FTP. Also, change HTTP to HTTPS for URLs to gnu.org and fsf.org when this works, as this will further help defend against man-in-the-middle attacks (for this part I omitted the MS-DOS and MS-Windows sources and the test tarballs to keep the workload down). HTTPS is not fully working to lists.gnu.org so I left those URLs alone for now.
331 lines
9.7 KiB
C
331 lines
9.7 KiB
C
/* Test GNU Emacs modules.
|
||
|
||
Copyright 2015-2017 Free Software Foundation, Inc.
|
||
|
||
This file is part of GNU Emacs.
|
||
|
||
GNU Emacs 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 3 of the License, or (at
|
||
your option) any later version.
|
||
|
||
GNU Emacs 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 GNU Emacs. If not, see <https://www.gnu.org/licenses/>. */
|
||
|
||
#include <assert.h>
|
||
#include <stdio.h>
|
||
#include <stdlib.h>
|
||
#include <emacs-module.h>
|
||
|
||
int plugin_is_GPL_compatible;
|
||
|
||
/* Always return symbol 't'. */
|
||
static emacs_value
|
||
Fmod_test_return_t (emacs_env *env, ptrdiff_t nargs, emacs_value args[],
|
||
void *data)
|
||
{
|
||
return env->intern (env, "t");
|
||
}
|
||
|
||
/* Expose simple sum function. */
|
||
static intmax_t
|
||
sum (intmax_t a, intmax_t b)
|
||
{
|
||
return a + b;
|
||
}
|
||
|
||
static emacs_value
|
||
Fmod_test_sum (emacs_env *env, ptrdiff_t nargs, emacs_value args[], void *data)
|
||
{
|
||
assert (nargs == 2);
|
||
|
||
intmax_t a = env->extract_integer (env, args[0]);
|
||
intmax_t b = env->extract_integer (env, args[1]);
|
||
|
||
intmax_t r = sum (a, b);
|
||
|
||
return env->make_integer (env, r);
|
||
}
|
||
|
||
|
||
/* Signal '(error 56). */
|
||
static emacs_value
|
||
Fmod_test_signal (emacs_env *env, ptrdiff_t nargs, emacs_value args[],
|
||
void *data)
|
||
{
|
||
assert (env->non_local_exit_check (env) == emacs_funcall_exit_return);
|
||
env->non_local_exit_signal (env, env->intern (env, "error"),
|
||
env->make_integer (env, 56));
|
||
return env->intern (env, "nil");
|
||
}
|
||
|
||
|
||
/* Throw '(tag 65). */
|
||
static emacs_value
|
||
Fmod_test_throw (emacs_env *env, ptrdiff_t nargs, emacs_value args[],
|
||
void *data)
|
||
{
|
||
assert (env->non_local_exit_check (env) == emacs_funcall_exit_return);
|
||
env->non_local_exit_throw (env, env->intern (env, "tag"),
|
||
env->make_integer (env, 65));
|
||
return env->intern (env, "nil");
|
||
}
|
||
|
||
|
||
/* Call argument function, catch all non-local exists and return
|
||
either normal result or a list describing the non-local exit. */
|
||
static emacs_value
|
||
Fmod_test_non_local_exit_funcall (emacs_env *env, ptrdiff_t nargs,
|
||
emacs_value args[], void *data)
|
||
{
|
||
assert (nargs == 1);
|
||
emacs_value result = env->funcall (env, args[0], 0, NULL);
|
||
emacs_value non_local_exit_symbol, non_local_exit_data;
|
||
enum emacs_funcall_exit code
|
||
= env->non_local_exit_get (env, &non_local_exit_symbol,
|
||
&non_local_exit_data);
|
||
switch (code)
|
||
{
|
||
case emacs_funcall_exit_return:
|
||
return result;
|
||
case emacs_funcall_exit_signal:
|
||
{
|
||
env->non_local_exit_clear (env);
|
||
emacs_value Flist = env->intern (env, "list");
|
||
emacs_value list_args[] = {env->intern (env, "signal"),
|
||
non_local_exit_symbol, non_local_exit_data};
|
||
return env->funcall (env, Flist, 3, list_args);
|
||
}
|
||
case emacs_funcall_exit_throw:
|
||
{
|
||
env->non_local_exit_clear (env);
|
||
emacs_value Flist = env->intern (env, "list");
|
||
emacs_value list_args[] = {env->intern (env, "throw"),
|
||
non_local_exit_symbol, non_local_exit_data};
|
||
return env->funcall (env, Flist, 3, list_args);
|
||
}
|
||
}
|
||
|
||
/* Never reached. */
|
||
return env->intern (env, "nil");;
|
||
}
|
||
|
||
|
||
/* Return a global reference. */
|
||
static emacs_value
|
||
Fmod_test_globref_make (emacs_env *env, ptrdiff_t nargs, emacs_value args[],
|
||
void *data)
|
||
{
|
||
/* Make a big string and make it global. */
|
||
char str[26 * 100];
|
||
for (int i = 0; i < sizeof str; i++)
|
||
str[i] = 'a' + (i % 26);
|
||
|
||
/* We don't need to null-terminate str. */
|
||
emacs_value lisp_str = env->make_string (env, str, sizeof str);
|
||
return env->make_global_ref (env, lisp_str);
|
||
}
|
||
|
||
|
||
/* Return a copy of the argument string where every 'a' is replaced
|
||
with 'b'. */
|
||
static emacs_value
|
||
Fmod_test_string_a_to_b (emacs_env *env, ptrdiff_t nargs, emacs_value args[],
|
||
void *data)
|
||
{
|
||
emacs_value lisp_str = args[0];
|
||
ptrdiff_t size = 0;
|
||
char * buf = NULL;
|
||
|
||
env->copy_string_contents (env, lisp_str, buf, &size);
|
||
buf = malloc (size);
|
||
env->copy_string_contents (env, lisp_str, buf, &size);
|
||
|
||
for (ptrdiff_t i = 0; i + 1 < size; i++)
|
||
if (buf[i] == 'a')
|
||
buf[i] = 'b';
|
||
|
||
return env->make_string (env, buf, size - 1);
|
||
}
|
||
|
||
|
||
/* Embedded pointers in lisp objects. */
|
||
|
||
/* C struct (pointer to) that will be embedded. */
|
||
struct super_struct
|
||
{
|
||
int amazing_int;
|
||
char large_unused_buffer[512];
|
||
};
|
||
|
||
/* Return a new user-pointer to a super_struct, with amazing_int set
|
||
to the passed parameter. */
|
||
static emacs_value
|
||
Fmod_test_userptr_make (emacs_env *env, ptrdiff_t nargs, emacs_value args[],
|
||
void *data)
|
||
{
|
||
struct super_struct *p = calloc (1, sizeof *p);
|
||
p->amazing_int = env->extract_integer (env, args[0]);
|
||
return env->make_user_ptr (env, free, p);
|
||
}
|
||
|
||
/* Return the amazing_int of a passed 'user-pointer to a super_struct'. */
|
||
static emacs_value
|
||
Fmod_test_userptr_get (emacs_env *env, ptrdiff_t nargs, emacs_value args[],
|
||
void *data)
|
||
{
|
||
struct super_struct *p = env->get_user_ptr (env, args[0]);
|
||
return env->make_integer (env, p->amazing_int);
|
||
}
|
||
|
||
|
||
/* Fill vector in args[0] with value in args[1]. */
|
||
static emacs_value
|
||
Fmod_test_vector_fill (emacs_env *env, ptrdiff_t nargs, emacs_value args[],
|
||
void *data)
|
||
{
|
||
emacs_value vec = args[0];
|
||
emacs_value val = args[1];
|
||
ptrdiff_t size = env->vec_size (env, vec);
|
||
for (ptrdiff_t i = 0; i < size; i++)
|
||
env->vec_set (env, vec, i, val);
|
||
return env->intern (env, "t");
|
||
}
|
||
|
||
|
||
/* Return whether all elements of vector in args[0] are 'eq' to value
|
||
in args[1]. */
|
||
static emacs_value
|
||
Fmod_test_vector_eq (emacs_env *env, ptrdiff_t nargs, emacs_value args[],
|
||
void *data)
|
||
{
|
||
emacs_value vec = args[0];
|
||
emacs_value val = args[1];
|
||
ptrdiff_t size = env->vec_size (env, vec);
|
||
for (ptrdiff_t i = 0; i < size; i++)
|
||
if (!env->eq (env, env->vec_get (env, vec, i), val))
|
||
return env->intern (env, "nil");
|
||
return env->intern (env, "t");
|
||
}
|
||
|
||
static emacs_value invalid_stored_value;
|
||
|
||
/* The next two functions perform a possibly-invalid operation: they
|
||
store a value in a static variable and load it. This causes
|
||
undefined behavior if the environment that the value was created
|
||
from is no longer live. The module assertions check for this
|
||
error. */
|
||
|
||
static emacs_value
|
||
Fmod_test_invalid_store (emacs_env *env, ptrdiff_t nargs, emacs_value *args,
|
||
void *data)
|
||
{
|
||
return invalid_stored_value = env->make_integer (env, 123);
|
||
}
|
||
|
||
static emacs_value
|
||
Fmod_test_invalid_load (emacs_env *env, ptrdiff_t nargs, emacs_value *args,
|
||
void *data)
|
||
{
|
||
return invalid_stored_value;
|
||
}
|
||
|
||
/* An invalid finalizer: Finalizers are run during garbage collection,
|
||
where Lisp code can’t be executed. -module-assertions tests for
|
||
this case. */
|
||
|
||
static emacs_env *current_env;
|
||
|
||
static void
|
||
invalid_finalizer (void *ptr)
|
||
{
|
||
current_env->intern (current_env, "nil");
|
||
}
|
||
|
||
static emacs_value
|
||
Fmod_test_invalid_finalizer (emacs_env *env, ptrdiff_t nargs, emacs_value *args,
|
||
void *data)
|
||
{
|
||
current_env = env;
|
||
env->make_user_ptr (env, invalid_finalizer, NULL);
|
||
return env->funcall (env, env->intern (env, "garbage-collect"), 0, NULL);
|
||
}
|
||
|
||
|
||
/* Lisp utilities for easier readability (simple wrappers). */
|
||
|
||
/* Provide FEATURE to Emacs. */
|
||
static void
|
||
provide (emacs_env *env, const char *feature)
|
||
{
|
||
emacs_value Qfeat = env->intern (env, feature);
|
||
emacs_value Qprovide = env->intern (env, "provide");
|
||
emacs_value args[] = { Qfeat };
|
||
|
||
env->funcall (env, Qprovide, 1, args);
|
||
}
|
||
|
||
/* Bind NAME to FUN. */
|
||
static void
|
||
bind_function (emacs_env *env, const char *name, emacs_value Sfun)
|
||
{
|
||
emacs_value Qfset = env->intern (env, "fset");
|
||
emacs_value Qsym = env->intern (env, name);
|
||
emacs_value args[] = { Qsym, Sfun };
|
||
|
||
env->funcall (env, Qfset, 2, args);
|
||
}
|
||
|
||
/* Module init function. */
|
||
int
|
||
emacs_module_init (struct emacs_runtime *ert)
|
||
{
|
||
if (ert->size < sizeof *ert)
|
||
{
|
||
fprintf (stderr, "Runtime size of runtime structure (%td bytes) "
|
||
"smaller than compile-time size (%zu bytes)",
|
||
ert->size, sizeof *ert);
|
||
return 1;
|
||
}
|
||
|
||
emacs_env *env = ert->get_environment (ert);
|
||
|
||
if (env->size < sizeof *env)
|
||
{
|
||
fprintf (stderr, "Runtime size of environment structure (%td bytes) "
|
||
"smaller than compile-time size (%zu bytes)",
|
||
env->size, sizeof *env);
|
||
return 2;
|
||
}
|
||
|
||
#define DEFUN(lsym, csym, amin, amax, doc, data) \
|
||
bind_function (env, lsym, \
|
||
env->make_function (env, amin, amax, csym, doc, data))
|
||
|
||
DEFUN ("mod-test-return-t", Fmod_test_return_t, 1, 1, NULL, NULL);
|
||
DEFUN ("mod-test-sum", Fmod_test_sum, 2, 2, "Return A + B\n\n(fn a b)", NULL);
|
||
DEFUN ("mod-test-signal", Fmod_test_signal, 0, 0, NULL, NULL);
|
||
DEFUN ("mod-test-throw", Fmod_test_throw, 0, 0, NULL, NULL);
|
||
DEFUN ("mod-test-non-local-exit-funcall", Fmod_test_non_local_exit_funcall,
|
||
1, 1, NULL, NULL);
|
||
DEFUN ("mod-test-globref-make", Fmod_test_globref_make, 0, 0, NULL, NULL);
|
||
DEFUN ("mod-test-string-a-to-b", Fmod_test_string_a_to_b, 1, 1, NULL, NULL);
|
||
DEFUN ("mod-test-userptr-make", Fmod_test_userptr_make, 1, 1, NULL, NULL);
|
||
DEFUN ("mod-test-userptr-get", Fmod_test_userptr_get, 1, 1, NULL, NULL);
|
||
DEFUN ("mod-test-vector-fill", Fmod_test_vector_fill, 2, 2, NULL, NULL);
|
||
DEFUN ("mod-test-vector-eq", Fmod_test_vector_eq, 2, 2, NULL, NULL);
|
||
DEFUN ("mod-test-invalid-store", Fmod_test_invalid_store, 0, 0, NULL, NULL);
|
||
DEFUN ("mod-test-invalid-load", Fmod_test_invalid_load, 0, 0, NULL, NULL);
|
||
DEFUN ("mod-test-invalid-finalizer", Fmod_test_invalid_finalizer, 0, 0,
|
||
NULL, NULL);
|
||
|
||
#undef DEFUN
|
||
|
||
provide (env, "mod-test");
|
||
return 0;
|
||
}
|