-
Notifications
You must be signed in to change notification settings - Fork 135
Process aliases #2027
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
base: main
Are you sure you want to change the base?
Process aliases #2027
Conversation
e2c4216 to
f30b1f3
Compare
|
@bettio it seems that CI sometimes fails because it cannot find |
src/libAtomVM/term.h
Outdated
| #define BOXED_FUN_SIZE 3 | ||
| #define FLOAT_SIZE (sizeof(float_term_t) / sizeof(term) + 1) | ||
| #define REF_SIZE ((int) ((sizeof(uint64_t) / sizeof(term)) + 1)) | ||
| #define TERM_BOXED_PID_REF_SIZE (REF_SIZE + 1) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
For consistency with the work we are doing lately I suggest renaming all PID_REF and pid_reference to PROCESS_REF and process_reference
src/libAtomVM/term.h
Outdated
| term *boxed_value = memory_heap_alloc(heap, TERM_BOXED_PID_REF_SIZE); | ||
| boxed_value[0] = TERM_BOXED_PID_REF_HEADER; | ||
|
|
||
| #if TERM_BYTES == 8 |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
nitpicking/very low priority comment: just for consistency I would suggest moving first the 4 branch:
#if TERM_BYTES == 4 ... #elif TERM_BYTES == 8.
Same applies to the other functions as well.
src/libAtomVM/term.c
Outdated
| return ret; | ||
|
|
||
| } else if (term_is_pid_reference(t)) { | ||
| int32_t process_id = term_pid_ref_to_process_id(t); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
We are using uint32_t for process id.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Are you sure? It's int32 here: https://github.com/atomvm/AtomVM/blob/main/src/libAtomVM/context.h#L106 and in many other places
tests/erlang_tests/test_monitor.erl
Outdated
| ok = test_alias(), | ||
| ok = test_monitor_alias(), | ||
| ok = test_monitor_alias_explicit_unalias(), | ||
| ok = test_monitor_down_alias(), |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
We should also test unhappy paths, such as:
- double/triple unalias
- alias a dead process
- unalias a dead process
In addition multiple aliases should be tested, this is legit but is should be properly tested.
src/libAtomVM/nifs.c
Outdated
| RAISE_ERROR(BADARG_ATOM); | ||
| } | ||
|
|
||
| while (!term_is_nil(options)) { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
We should use term_is_nonempty_list to avoid issues with non proper lists.
src/libAtomVM/nifs.c
Outdated
| RAISE_ERROR(UNSUPPORTED_ATOM); | ||
| } else { | ||
| RAISE_ERROR(BADARG_ATOM); | ||
| } |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
here we check that last list item is nil.
src/libAtomVM/context.h
Outdated
| enum ContextMonitorAliasType | ||
| { | ||
| CONTEXT_MONITOR_ALIAS_EXPLICIT_UNALIAS, | ||
| CONTEXT_MONITOR_ALIAS_DEMONITOR, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Let's use PascalCase. See AVMCCS-N004 rule in our coding style guide.
(There is still some old code that has to be migrated to updated style).
4b23228 to
07373d0
Compare
|
@bettio added more tests and missing features. The CI is failing, but it seems unrelated to the changes |
bettio
left a comment
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Looks mostly good: there is only one change related to IS_NULL_PTR that is required. I suggest also rebasing on top of main, since main has a lot of fixes for flaky tests.
src/libAtomVM/opcodesswitch.h
Outdated
| Context *p = globalcontext_get_process_lock(glb, local_process_id); | ||
| if (p) { | ||
| struct MonitorAlias *alias = context_find_alias(p, ref_ticks); | ||
| if (!IS_NULL_PTR(alias)) { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
We should not use !IS_NULL_PTR (x) since it is a shorthand for UNLIKELY(x == NULL). So we must use it only when it is unlikely that the pointer is null (that is mostly for error handling purposes).
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Done, but there are many occurrences of !IS_NULL_PTR in the code
92db4b7 to
1bdef14
Compare
|
esp32 tests are crashing |
pguyot
left a comment
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
My main concern so far is that monitors are now extended to 2 or 3 words instead of 1 or 2. This may break some code that took monitors and passed it around as references, as the term_to_ref_ticks/term_from_ref_ticks is a common pattern. A similar breakage existed with resources being larger references if one tries to pass a resource where a 1/2 words ref is expected, but it is less likely as resources were not references so far. This may be the crash cause of esp32 tests.
port.erl
call(Port, Message, Timeout) ->
MonitorRef = monitor(port, Port),
Port ! {'$call', {self(), MonitorRef}, Message},
Result =
receive
{'DOWN', MonitorRef, port, Port, normal} ->
{error, noproc};
{'DOWN', MonitorRef, port, Port, Reason} ->
{error, Reason};
out_of_memory ->
out_of_memory;
{MonitorRef, Ret} ->
Ret
after Timeout ->
{error, timeout}
end,
demonitor(MonitorRef, [flush]),
Result.
uart_driver.c:
static void uart_driver_do_read(Context *ctx, GenMessage gen_message)
{
GlobalContext *glb = ctx->global;
struct UARTData *uart_data = ctx->platform_data;
term pid = gen_message.pid;
term ref = gen_message.ref;
uint64_t ref_ticks = term_to_ref_ticks(ref);
| -type raise_stacktrace() :: | ||
| [{module(), atom(), arity() | [term()]} | {function(), arity() | [term()]}] | stacktrace(). | ||
|
|
||
| -type monitor_option() :: {'alias', 'explicit_unalias' | 'demonitor' | 'reply_demonitor'}. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This is not enforced by style guide but usually we don't quote atoms when it's not required.
| } | ||
| // Prepare the message on ctx's heap which will be freed afterwards. | ||
| term ref = term_from_ref_ticks(monitored_monitor->ref_ticks, &ctx->heap); | ||
| term ref = term_make_process_reference(target->process_id, monitored_monitor->ref_ticks, &ctx->heap); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This is not what OTP does and there probably is a good reason.
Regular monitors are smaller than aliases.
7> Pid = spawn(fun() -> receive ok -> ok end end).
<0.94.0>
8> monitor(process, Pid).
#Ref<0.3008598808.1200095248.57512>
9> monitor(process, Pid, [{alias, explicit_unalias}]).
#Ref<0.0.11651.3008598808.1200160784.57533>
10> make_ref().
#Ref<0.3008598808.1200095248.57553>
src/libAtomVM/term.h
Outdated
| boxed_value[0] = TERM_BOXED_PROCESS_REF_HEADER; | ||
|
|
||
| #if TERM_BYTES == 4 | ||
| boxed_value[1] = (ref_ticks >> 4); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This was fixed for regular refs, we probably want ref_ticks >> 32
| end, | ||
| Term. | ||
|
|
||
| is_atomvm_or_otp_version_at_least(OTPVersion) -> |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
get_otp_version/0 already exists in this module.
Also there is a poor usage (fixed in #2025 )
OTPVersion = get_otp_version(),
if
OTPVersion =:= atomvm orelse OTPVersion >= 26 ->
B == erlang:term_to_binary(A);
true ->
true
end.
be sure to rather do:
OTPVersion = get_otp_version(),
if
OTPVersion >= 26 ->
B == erlang:term_to_binary(A);
true ->
true
end.
as atoms sort higher than integers.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Also there is a poor usage
I would call it 'a readable way' :P
Signed-off-by: Mateusz Front <[email protected]>
ce57282 to
eac7682
Compare
|
@pguyot the reason for the STM32 tests failing was that the size of a regular reference is actually 3 terms there, which means process reference takes 4 terms, and it probably conflicts with resource reference size, which is always 4 terms. I changed the process reference size to 5, but that's not a perfect solution :P So we need to come up with another way of distinguishing references - do you have anything in mind? I also added |
Signed-off-by: Mateusz Front <[email protected]>
eac7682 to
68ccb22
Compare
TODO:
spawn_optThese changes are made under both the "Apache 2.0" and the "GNU Lesser General
Public License 2.1 or later" license terms (dual license).
SPDX-License-Identifier: Apache-2.0 OR LGPL-2.1-or-later