From a85a45b5a384deae60f8020d0805a93df141a7c3 Mon Sep 17 00:00:00 2001
From: Sverker Eriksson
+typedef struct {
+ ErlNifResourceDtor* dtor;
+ ErlNifResourceStop* stop;
+} ErlNifResourceTypeInit;
+ Initialization structure read by
typedef void ErlNifResourceDtor(ErlNifEnv* env, void* obj);
The function prototype of a resource destructor function.
+typedef void ErlNifResourceStop(ErlNifEnv* env, void* obj);
+ The function prototype of a resource stop function,
+ called on the behalf of
@@ -2238,6 +2256,24 @@ enif_map_iterator_destroy(env, &iter);
+ Same as
Argument
Removes a reference to resource object
Removes a reference to resource object
This function can be used to receive asynchronous notifications + when OS-specific event objects become ready for either read or write operations.
+Argument
Argument
{select, Obj, Ref, ready_input | ready_output}
+ Argument
Argument
The notifications are one-shot only. To receive further notifications of the same
+ type (read or write), repeated calls to
Use
Returns 0 on success, or -1 if invalid arguments.
+
-typedef void ErlNifResourceStop(ErlNifEnv* env, void* obj);
+typedef void ErlNifResourceStop(ErlNifEnv* env, void* obj, ErlNifEvent event, int is_direct_call);
The function prototype of a resource stop function,
called on the behalf of
Returns 0 on success, or -1 if invalid arguments.
+Returns an integer where different bits indicate the outcome of the call:
+The return value from a successful call with
Always use bitwise AND to test the return value. New significant bits
+ may be added in future releases to give more detailed information for both
+ failed and successful calls. Do NOT use equallity tests like
Example:
+
+retval = enif_select(env, fd, ERL_NIF_SELECT_READ, resource, ref);
+if (retval & ERL_NIF_SELECT_ERROR) {
+ /* handle error */
+}
+/* Success! */
+
+
typedef void ErlNifResourceDtor(ErlNifEnv* env, void* obj);
The function prototype of a resource destructor function.
+The
+typedef void ErlNifResourceDown(ErlNifEnv* env, void* obj, const ErlNifPid* pid, const ErlNifMonitor* mon);
+ The function prototype of a resource down function,
+ called on the behalf of
Cancels a monitor created earlier with
Returns
This function is only thread-safe when the emulator with SMP support + is used. It can only be used in a non-SMP emulator from a NIF-calling + thread.
+Starts monitoring a process from a resource. When a process is
+ monitored, a process exit results in a call to the provided
+
Argument
If
Returns
This function is only thread-safe when the emulator with SMP support + is used. It can only be used in a non-SMP emulator from a NIF-calling + thread.
+Argument
{select, Obj, Ref, ready_input | ready_output}
Argument
Argument
The notifications are one-shot only. To receive further notifications of the same
- type (read or write), repeated calls to
Use
Returns an integer where different bits indicate the outcome of the call:
+Returns a non-negative value on success where the following bits can be set:
Returns a negative value if the call failed where the follwing bits can be set:
+The return value from a successful call with
Always use bitwise AND to test the return value. New significant bits
- may be added in future releases to give more detailed information for both
- failed and successful calls. Do NOT use equallity tests like
Use bitwise AND to test for specific bits in the return vaue.
+ New significant bits may be added in future releases to give more detailed
+ information for both failed and successful calls. Do NOT use equallity tests
+ like
Example:
-retval = enif_select(env, fd, ERL_NIF_SELECT_READ, resource, ref);
-if (retval & ERL_NIF_SELECT_ERROR) {
+retval = enif_select(env, fd, ERL_NIF_SELECT_STOP, resource, ref);
+if (retval < 0) {
/* handle error */
}
/* Success! */
+if (retval & ERL_NIF_SELECT_STOP_CALLED) {
+ /* ... */
+}
Example:
retval = enif_select(env, fd, ERL_NIF_SELECT_STOP, resource, ref);
-if (retval < 0) {
+if (retval < 0) {
/* handle error */
}
/* Success! */
--
cgit v1.2.3
From 839ad04bbacb329c89568371bbf5a28d4b2bba25 Mon Sep 17 00:00:00 2001
From: Sverker Eriksson
Date: Tue, 14 Feb 2017 14:23:15 +0100
Subject: Fix whitebox monitor tests
---
erts/emulator/test/erl_link_SUITE.erl | 4 ++--
1 file changed, 2 insertions(+), 2 deletions(-)
diff --git a/erts/emulator/test/erl_link_SUITE.erl b/erts/emulator/test/erl_link_SUITE.erl
index 89e1aefb50..9258897764 100644
--- a/erts/emulator/test/erl_link_SUITE.erl
+++ b/erts/emulator/test/erl_link_SUITE.erl
@@ -60,7 +60,7 @@
% These are to be kept in sync with erl_monitors.h
-define(MON_ORIGIN, 1).
--define(MON_TARGET, 3).
+-define(MON_TARGET, 2).
-record(erl_link, {type = ?LINK_UNDEF,
@@ -69,7 +69,7 @@
% This is to be kept in sync with erl_bif_info.c (make_monitor_list)
--record(erl_monitor, {type, % MON_ORIGIN or MON_TARGET (1 or 3)
+-record(erl_monitor, {type, % MON_ORIGIN or MON_TARGET
ref,
pid, % Process or nodename
name = []}). % registered name or []
--
cgit v1.2.3
From 28e6aeb8c0cc0e8b8821be957e3ee9c1eb015de6 Mon Sep 17 00:00:00 2001
From: Sverker Eriksson
Date: Tue, 14 Feb 2017 16:41:19 +0100
Subject: Fix enif_select for windows
---
erts/emulator/beam/erl_nif.h | 8 +++++---
erts/emulator/sys/common/erl_check_io.c | 3 ++-
2 files changed, 7 insertions(+), 4 deletions(-)
diff --git a/erts/emulator/beam/erl_nif.h b/erts/emulator/beam/erl_nif.h
index 9ff28a30cc..ac45f3ac81 100644
--- a/erts/emulator/beam/erl_nif.h
+++ b/erts/emulator/beam/erl_nif.h
@@ -138,9 +138,11 @@ typedef struct
void* ref_bin;
}ErlNifBinary;
-//#ifndef ERL_SYS_DRV
-typedef int ErlNifEvent; /* An event to be selected on. */
-//#endif
+#if (defined(__WIN32__) || defined(_WIN32) || defined(_WIN32_))
+typedef void* ErlNifEvent; /* FIXME: Use 'HANDLE' somehow without breaking existing source */
+#else
+typedef int ErlNifEvent;
+#endif
/* Return bits from enif_select: */
#define ERL_NIF_SELECT_STOP_CALLED (1 << 0)
diff --git a/erts/emulator/sys/common/erl_check_io.c b/erts/emulator/sys/common/erl_check_io.c
index 1c97df4201..8191b6e53f 100644
--- a/erts/emulator/sys/common/erl_check_io.c
+++ b/erts/emulator/sys/common/erl_check_io.c
@@ -201,7 +201,8 @@ static ERTS_INLINE ErtsDrvEventState* hash_new_drv_ev_state(ErtsSysFdType fd)
#if ERTS_CIO_HAVE_DRV_EVENT
tmpl.driver.event = NULL;
#endif
- tmpl.driver.drv_ptr = NULL;
+ tmpl.driver.nif = NULL;
+ tmpl.driver.stop.drv_ptr = NULL;
tmpl.events = 0;
tmpl.remove_cnt = 0;
tmpl.type = ERTS_EV_TYPE_NONE;
--
cgit v1.2.3
From d7abb5e2efc5dfd0f100b9a8978bc9e87e15cd2d Mon Sep 17 00:00:00 2001
From: Sverker Eriksson
Date: Thu, 16 Feb 2017 14:42:24 +0100
Subject: erts: Skip nif_SUITE:select on windows
for now...
---
erts/emulator/test/nif_SUITE.erl | 7 +++++++
erts/emulator/test/nif_SUITE_data/nif_SUITE.c | 15 ++++++++++++---
2 files changed, 19 insertions(+), 3 deletions(-)
diff --git a/erts/emulator/test/nif_SUITE.erl b/erts/emulator/test/nif_SUITE.erl
index 8439b28010..c1d9632fb5 100644
--- a/erts/emulator/test/nif_SUITE.erl
+++ b/erts/emulator/test/nif_SUITE.erl
@@ -128,6 +128,13 @@ init_per_testcase(hipe, Config) ->
undefined -> {skip, "HiPE is disabled"};
_ -> Config
end;
+init_per_testcase(select, Config) ->
+ case os:type() of
+ {win32,_} ->
+ {skip, "Test not yet implemented for windows"};
+ _ ->
+ Config
+ end;
init_per_testcase(_Case, Config) ->
Config.
diff --git a/erts/emulator/test/nif_SUITE_data/nif_SUITE.c b/erts/emulator/test/nif_SUITE_data/nif_SUITE.c
index 4f003d11f5..69c5f09bd5 100644
--- a/erts/emulator/test/nif_SUITE_data/nif_SUITE.c
+++ b/erts/emulator/test/nif_SUITE_data/nif_SUITE.c
@@ -23,10 +23,10 @@
#include
#include
#include
+#include
#ifndef __WIN32__
#include
#include
-#include
#endif
#include "nif_mod.h"
@@ -143,7 +143,7 @@ static ErlNifResourceTypeInit fd_rt_init = {
fd_resource_stop
};
struct fd_resource {
- int fd;
+ ErlNifEvent fd;
int was_selected;
ErlNifPid pid;
};
@@ -2144,7 +2144,7 @@ static ERL_NIF_TERM select_nif(ErlNifEnv* env, int argc, const ERL_NIF_TERM argv
int retval;
if (!get_fd(env, argv[0], &fdr)
- || !enif_get_uint(env, argv[1], &mode)
+ || !enif_get_uint(env, argv[1], (unsigned int*)&mode)
|| !enif_get_resource(env, argv[2], fd_resource_type, &obj))
{
return enif_make_badarg(env);
@@ -2164,6 +2164,7 @@ static ERL_NIF_TERM select_nif(ErlNifEnv* env, int argc, const ERL_NIF_TERM argv
return enif_make_int(env, retval);
}
+#ifndef __WIN32__
static ERL_NIF_TERM pipe_nif(ErlNifEnv* env, int argc, const ERL_NIF_TERM argv[])
{
struct fd_resource* read_rsrc;
@@ -2274,15 +2275,21 @@ static ERL_NIF_TERM is_closed_nif(ErlNifEnv* env, int argc, const ERL_NIF_TERM a
return fdr->fd < 0 ? atom_true : atom_false;
}
+#endif /* !__WIN32__ */
+
static void fd_resource_dtor(ErlNifEnv* env, void* obj)
{
struct fd_resource* fdr = (struct fd_resource*)obj;
resource_dtor(env, obj);
+#ifdef __WIN32__
+ abort();
+#else
if (fdr->fd >= 0) {
assert(!fdr->was_selected);
close(fdr->fd);
}
+#endif
}
static struct {
@@ -2924,10 +2931,12 @@ static ErlNifFunc nif_funcs[] =
{"port_command_nif", 2, port_command},
{"format_term_nif", 2, format_term},
{"select_nif", 5, select_nif},
+#ifndef __WIN32__
{"pipe_nif", 0, pipe_nif},
{"write_nif", 2, write_nif},
{"read_nif", 2, read_nif},
{"is_closed_nif", 1, is_closed_nif},
+#endif
{"last_fd_stop_call", 0, last_fd_stop_call},
{"alloc_monitor_resource_nif", 0, alloc_monitor_resource_nif},
{"monitor_process_nif", 4, monitor_process_nif},
--
cgit v1.2.3
From d8d8301a252579a000b24bab87d26549da0e813a Mon Sep 17 00:00:00 2001
From: Sverker Eriksson
Date: Fri, 17 Feb 2017 17:54:03 +0100
Subject: Remove faulty debug ASSERT
Why did I add that?
---
erts/emulator/sys/common/erl_poll.c | 1 -
1 file changed, 1 deletion(-)
diff --git a/erts/emulator/sys/common/erl_poll.c b/erts/emulator/sys/common/erl_poll.c
index 3cb0eb31f5..5e7ae8953a 100644
--- a/erts/emulator/sys/common/erl_poll.c
+++ b/erts/emulator/sys/common/erl_poll.c
@@ -3029,7 +3029,6 @@ ERTS_POLL_EXPORT(erts_poll_get_selected_events)(ErtsPollSet ps,
ev[fd] = 0;
else {
ev[fd] = ps->fds_status[fd].events;
- ASSERT(ps->fds_status[fd].used_events == ev[fd]);
if (
#if ERTS_POLL_USE_WAKEUP_PIPE
fd == ps->wake_fds[0] || fd == ps->wake_fds[1] ||
--
cgit v1.2.3
From af7cf70ca22a34add7836963d086ca0764f4fbae Mon Sep 17 00:00:00 2001
From: Sverker Eriksson
Date: Mon, 20 Feb 2017 20:20:29 +0100
Subject: Fix ErlNifMonitor handling
---
erts/emulator/beam/erl_nif.c | 7 ++-----
erts/emulator/beam/global.h | 1 +
erts/emulator/beam/io.c | 6 +++---
erts/emulator/test/nif_SUITE_data/nif_SUITE.c | 4 ++--
4 files changed, 8 insertions(+), 10 deletions(-)
diff --git a/erts/emulator/beam/erl_nif.c b/erts/emulator/beam/erl_nif.c
index daec4ac9e9..e6da4c1a76 100644
--- a/erts/emulator/beam/erl_nif.c
+++ b/erts/emulator/beam/erl_nif.c
@@ -2305,9 +2305,6 @@ void erts_resource_stop(ErtsResource* resource, ErlNifEvent e,
post_nif_noproc(&msg_env);
}
-/* SVERK SVERK: Move to ?.h */
-void erts_ref_to_driver_monitor(Eterm ref, ErlDrvMonitor*);
-
void erts_fire_nif_monitor(ErtsResource* resource, Eterm pid, Eterm ref)
{
ErtsMonitor* rmon;
@@ -3207,8 +3204,8 @@ int enif_demonitor_process(ErlNifEnv* env, void* obj, const ErlNifMonitor* monit
execution_state(env, NULL, &scheduler);
- memcpy(ref_heap, monitor, sizeof(Eterm)*ERTS_REF_THING_SIZE);
- ref = make_internal_ref(ref_heap);
+ ref = erts_driver_monitor_to_ref(ref_heap, monitor);
+
erts_smp_mtx_lock(&rsrc->monitors->lock);
mon = erts_remove_monitor(&rsrc->monitors->root, ref);
diff --git a/erts/emulator/beam/global.h b/erts/emulator/beam/global.h
index 776f2c599b..c4c848f49f 100644
--- a/erts/emulator/beam/global.h
+++ b/erts/emulator/beam/global.h
@@ -1176,6 +1176,7 @@ void erts_stale_drv_select(Eterm, ErlDrvPort, ErlDrvEvent, int, int);
Port *erts_get_heart_port(void);
void erts_emergency_close_ports(void);
void erts_ref_to_driver_monitor(Eterm ref, ErlDrvMonitor *mon);
+Eterm erts_driver_monitor_to_ref(Eterm* hp, const ErlDrvMonitor *mon);
#if defined(ERTS_SMP) && defined(ERTS_ENABLE_LOCK_COUNT)
void erts_lcnt_enable_io_lock_count(int enable);
diff --git a/erts/emulator/beam/io.c b/erts/emulator/beam/io.c
index 9b525cc100..84dc00f641 100644
--- a/erts/emulator/beam/io.c
+++ b/erts/emulator/beam/io.c
@@ -7615,7 +7615,7 @@ void erts_ref_to_driver_monitor(Eterm ref, ErlDrvMonitor *mon)
ERTS_REF_THING_SIZE*sizeof(Uint));
}
-static Eterm driver_monitor_to_ref(Eterm *hp, const ErlDrvMonitor *mon)
+Eterm erts_driver_monitor_to_ref(Eterm *hp, const ErlDrvMonitor *mon)
{
Eterm ref;
ERTS_CT_ASSERT(ERTS_REF_THING_SIZE*sizeof(Uint) <= sizeof(ErlDrvMonitor));
@@ -7685,7 +7685,7 @@ static int do_driver_demonitor_process(Port *prt, const ErlDrvMonitor *monitor)
ErtsMonitor *mon;
Eterm to;
- ref = driver_monitor_to_ref(heap, monitor);
+ ref = erts_driver_monitor_to_ref(heap, monitor);
mon = erts_lookup_monitor(ERTS_P_MONITORS(prt), ref);
if (mon == NULL) {
@@ -7742,7 +7742,7 @@ static ErlDrvTermData do_driver_get_monitored_process(Port *prt,const ErlDrvMoni
Eterm to;
Eterm heap[ERTS_REF_THING_SIZE];
- ref = driver_monitor_to_ref(heap, monitor);
+ ref = erts_driver_monitor_to_ref(heap, monitor);
mon = erts_lookup_monitor(ERTS_P_MONITORS(prt), ref);
if (mon == NULL) {
diff --git a/erts/emulator/test/nif_SUITE_data/nif_SUITE.c b/erts/emulator/test/nif_SUITE_data/nif_SUITE.c
index 69c5f09bd5..8fe5ee809a 100644
--- a/erts/emulator/test/nif_SUITE_data/nif_SUITE.c
+++ b/erts/emulator/test/nif_SUITE_data/nif_SUITE.c
@@ -2708,7 +2708,7 @@ static ERL_NIF_TERM monitor_frenzy_nif(ErlNifEnv* env, int argc, const ERL_NIF_T
else {
unsigned int resource_op = rand_bits(&rnd, 3);
r = resv[rix].obj;
- if (resource_op == 0) {
+ if (resource_op == 0) { /* delete resource */
resv[rix].obj = NULL;
resv[rix].release_cnt++;
enif_mutex_unlock(resv[rix].lock);
@@ -2717,7 +2717,7 @@ static ERL_NIF_TERM monitor_frenzy_nif(ErlNifEnv* env, int argc, const ERL_NIF_T
retval = atom_ok;
break;
}
- else if (resource_op == 1) {
+ else if (resource_op == 1) { /* return resource */
retval = enif_make_resource(env, r);
enif_mutex_unlock(resv[rix].lock);
break;
--
cgit v1.2.3
From 9c0ab3723c4cfb7f2d9061b661ae6a0331e72c8d Mon Sep 17 00:00:00 2001
From: Sverker Eriksson
Date: Wed, 22 Feb 2017 12:01:37 +0100
Subject: Add docs for enif_compare_monitors and ErlNifMonitor
---
erts/doc/src/erl_nif.xml | 43 ++++++++++++++++++++++++++++++++++++++-----
1 file changed, 38 insertions(+), 5 deletions(-)
diff --git a/erts/doc/src/erl_nif.xml b/erts/doc/src/erl_nif.xml
index b31052287d..b0a632d2d6 100644
--- a/erts/doc/src/erl_nif.xml
+++ b/erts/doc/src/erl_nif.xml
@@ -675,6 +675,18 @@ typedef struct {
When receiving data from untrusted sources, use option
ERL_NIF_BIN2TERM_SAFE .
This is an opaque data type that identifies a monitor.
+The nif writer is to provide the memory for storing the
+ monitor when calling
A process identifier (pid). In contrast to pid terms (instances of
@@ -909,6 +921,21 @@ typedef enum {
+ Compares two Returns Cancels a monitor created earlier with Cancels a monitor created earlier with Returns Starts monitoring a process from a resource. When a process is
monitored, a process exit results in a call to the provided
- Argument If Returns
@@ -2182,14 +2210,19 @@ enif_map_iterator_destroy(env, &iter);