12 Processes
12.1 Processes
Erlang is designed for massive concurrency. Erlang processes are lightweight (grow and shrink dynamically) with small memory footprint, fast to create and terminate, and the scheduling overhead is low.
12.2 Process Creation
A process is created by calling spawn()
:
spawn(Module, Name, Args) -> pid() Module = Name = atom() Args = [Arg1,...,ArgN] ArgI = term()
spawn()
creates a new process and returns the pid.
The new process starts executing in Module:Name(Arg1,...,ArgN)
where the arguments are the elements of the (possible empty) Args
argument list.
There exist a number of different spawn
BIFs:
spawn/1,2,3,4
spawn_link/1,2,3,4
spawn_monitor/1,2,3,4
spawn_opt/2,3,4,5
spawn_request/1,2,3,4,5
12.3 Registered Processes
Besides addressing a process by using its pid, there are also BIFs for registering a process under a name. The name must be an atom and is automatically unregistered if the process terminates:
BIF | Description |
register(Name, Pid) | Associates the name Name , an atom, with the process Pid . |
registered() | Returns a list of names that have been registered using register/2 . |
whereis(Name) | Returns the pid registered under Name , or undefined if the name is not registered. |
12.4 Process Aliases
When sending a message to a process, the receiving process can be identified by a PID
, a registered name
, or a process alias which is a term of the type reference
. The typical use case that process aliases were designed for is a request/reply scenario. Using a process alias when sending the reply makes it possible for the receiver of the reply to prevent the reply from reaching its message queue if the operation times out or if the connection between the processes is lost.
A process alias can be used as identifier of the receiver when sending a message using the send operator !
or send BIFs such as erlang:send/2
. As long as the process alias is active, messages will be delivered the same way as if the process identifier of the process that created the alias had been used. When the alias has been deactivated, messages sent using the alias will be dropped before entering the message queue of the receiver. Note that messages that at deactivation time already have entered the message queue will not be removed.
A process alias is created either by calling one of the alias/0,1
BIFs or by creating an alias and a monitor simultaneously. If the alias is created together with a monitor, the same reference will be used both as monitor reference and alias. Creating a monitor and an alias at the same time is done by passing the {alias, _}
option to the monitor/3
BIF. The {alias, _}
option can also be passed when creating a monitor via spawn_opt()
, or spawn_request()
.
A process alias can be deactivated by the process that created it by calling the unalias/1
BIF. It is also possible to automatically deactivate an alias on certain events. See the documentation of the alias/1
BIF, and the {alias, _}
option of the monitor/3
BIF for more information about automatic deactivation of aliases.
It is not possible to:
- create an alias identifying another process than the caller.
- deactivate an alias unless it identifies the caller.
- look up an alias.
- look up the process identified by an alias.
- check if an alias is active or not.
- check if a reference is an alias.
These are all intentional design decisions relating to performance, scalability, and distribution transparency.
12.5 Process Termination
When a process terminates, it always terminates with an exit reason. The reason can be any term.
A process is said to terminate normally, if the exit reason is the atom normal
. A process with no more code to execute terminates normally.
A process terminates with an exit reason {Reason,Stack}
when a run-time error occurs. See Exit Reasons
.
A process can terminate itself by calling one of the following BIFs:
exit(Reason)
erlang:error(Reason)
erlang:error(Reason, Args)
The process then terminates with reason Reason
for exit/1
or {Reason,Stack}
for the others.
A process can also be terminated if it receives an exit signal with another exit reason than normal
, see Error Handling
.
12.6 Signals
All communication between Erlang processes and Erlang ports is done by sending and receiving asynchronous signals. The most common signals are Erlang message signals. A message signal can be sent using the send operator !
. A received message can be fetched from the message queue by the receiving process using the receive
expression.
Synchronous communication can be broken down into multiple asynchronous signals. An example of such a synchronous communication is a call to the erlang:process_info/2
BIF when the first argument does not equal the process identifier of the calling process. The caller sends an asynchronous signal requesting information, and then blocks waiting for the reply signal containing the requested information. When the request signal reaches its destination, the destination process replies with the requested information.
Sending Signals
There are many signals that processes and ports use to communicate. The list below contains the most important signals. In all the cases of request/reply signal pairs, the request signal is sent by the process calling the specific BIF, and the reply signal is sent back to it when the requested operation has been performed.
message
- Sent when using the
send operator !
, or when calling one of theerlang:send/2,3
orerlang:send_nosuspend/2,3
BIFs. link
- Sent when calling the
link/1
BIF. unlink
- Sent when calling the
unlink/1
BIF. exit
- Sent either when explicitly sending an
exit
signal by calling theexit/2
BIF, or when alinked process terminates
. If the signal is sent due to a link, the signal is sent after alldirectly visible Erlang resources
used by the process have been released. monitor
- Sent when calling one of the
monitor/2,3
BIFs. demonitor
- Sent when calling one of the
demonitor/1,2
BIFs, or when a process monitoring another process terminates. down
- Sent by a
monitored process or port that terminates
. The signal is sent after alldirectly visible Erlang resources
used by the process or the port have been released. change
- Sent by the
clock service
on the local runtime system, when thetime offset
changes, to processes which havemonitored the time_offset
. group_leader
- Sent when calling the
group_leader/2
BIF. -
spawn_request
/spawn_reply
,open_port_request
/open_port_reply
- Sent due to a call to one of the
spawn/1,2,3,4
,spawn_link/1,2,3,4
,spawn_monitor/1,2,3,4
,spawn_opt/2,3,4,5
,spawn_request/1,2,3,4,5
, orerlang:open_port/2
BIFs. The request signal is sent to thespawn service
which responds with the reply signal. alive_request
/alive_reply
- Sent due to a call to the
is_process_alive/1
BIF. -
garbage_collect_request
/garbage_collect_reply
,check_process_code_request
/check_process_code_reply
,process_info_request
/process_info_reply
- Sent due to a call to one of the
garbage_collect/1,2
,erlang:check_process_code/2,3
, orprocess_info/1,2
BIFs. Note that if the request is directed towards the caller itself and it is a synchronous request, no signaling will be performed and the caller will instead synchronously perform the request before returning from the BIF. port_command
,port_connect
,port_close
- Sent by a process to a port on the local node using the
send operator !
, or by calling one of thesend()
BIFs. The signal is sent by passing a term on the format{Owner, {command, Data}}
,{Owner, {connect, Pid}}
, or{Owner, close}
as message. -
port_command_request
/port_command_reply
,port_connect_request
/port_connect_reply
,port_close_request
/port_close_reply
,port_control_request
/port_control_reply
,port_call_request
/port_call_reply
,port_info_request
/port_info_reply
- Sent due to a call to one of the
erlang:port_command/2,3
,erlang:port_connect/2
,erlang:port_close/1
,erlang:port_control/3
,erlang:port_call/3
,erlang:port_info/1,2
BIFs. The request signal is sent to a port on the local node which responds with the reply signal. -
timer_start_request
/timer_start_reply
,timer_cancel_request
/timer_cancel_reply
- Sent due to a call to one of the
erlang:send_after/3,4
,erlang:start_timer/3,4
, orerlang:cancel_timer/1,2
BIFs. The request signal is sent to thetimer service
which responds with the reply signal.
The clock service, the timer service, and the spawn service mentioned above are services provided by the runtime system. Each of these services consists of multiple independently executing entities. Such a service can be viewed as a group of processes, and could actually be implemented like that. Since each service consists of multiple independently executing entities, the order between multiple signals sent from one service to one process is not preserved. Note that this does not violate the signal ordering guarantee
of the language.
The realization of the signals described above may change both at runtime and due to changes in implementation. You may be able to detect such changes using receive
tracing or by inspecting message queues. However, these are internal implementation details of the runtime system that you should not rely on. As an example, many of the reply signals above are ordinary message signals. When the operation is synchronous, the reply signals do not have to be message signals. The current implementation takes advantage of this and, depending on the state of the system, use alternative ways of delivering the reply signals. The implementation of these reply signals may also, at any time, be changed to not use message signals where it previously did.
Receiving Signals
When a signal is received by a process, some kind of action is taken. The specific action taken depends on the signal type, contents of the signal, and the state of the receiving process. Actions taken for the most common signals:
message
- If the message signal was sent using a
process alias
that is no longer active, the message signal will be dropped; otherwise, if the alias is still active or the message signal was sent by other means, the message is added to the end of the message queue. When the message has been added to the message queue, the receiving process can fetch the message from the message queue using thereceive
expression. link
,unlink
- Very simplified it can be viewed as updating process local information about the link. A detailed description of the
link protocol
can be found in the Distribution Protocol chapter of the ERTS User's Guide. exit
- Set the receiver in an exiting state, drop the signal, or convert the signal into a message and add it to the end of the message queue. If the receiver is set in an exiting state, no more Erlang code will be executed and the process is scheduled for termination. The section
Receiving Exit Signals
below gives more details on the action taken when anexit
signal is received. monitor
,demonitor
- Update process local information about the monitor.
down
,change
- Convert into a message if the corresponding monitor is still active; otherwise, drop the signal. If the signal is converted into a message, it is also added to the end of the message queue.
group_leader
- Change the group leader of the process.
spawn_reply
- Convert into a message, or drop the signal depending on the reply and how the
spawn_request
signal was configured. If the signal is converted into a message it is also added to the end of the message queue. For more information see thespawn_request()
BIF. alive_request
- Schedule execution of the is alive test. If the process is in an exiting state, the is alive test will not be executed until after all
directly visible Erlang resources
used by the process have been released. Thealive_reply
will be sent after the is alive test has executed. -
process_info_request
,garbage_collect_request
,check_process_code_request
- Schedule execution of the requested operation. The reply signal will be sent when the operation has been executed.
Note that some actions taken when a signal is received involves scheduling further actions which will result in a reply signal when these scheduled actions have completed. This implies that the reply signals may be sent in a different order than the order of the incoming signals that triggered these operations. This does, however, not violate the signal ordering guarantee
of the language.
The order of messages in the message queue of a process reflects the order in which the signals corresponding to the messages has been received since all signals that add messages to the message queue add them at the end of the message queue
. Messages corresponding to signals from the same sender are also ordered in the same order as the signals were sent due to the signal ordering guarantee
of the language.
Directly Visible Erlang Resources
As described above, exit
signals due to links, down
signals, and reply signals from an exiting process due to alive_request
s are not sent until all directly visible Erlang resources held by the terminating process have been released. With directly visible Erlang resources we here mean all resources made available by the language excluding resources held by heap data, dirty native code execution and the process identifier of the terminating process. Examples of directly visible Erlang resources are registered name
and ETS
tables.
The Excluded Resources
The process identifier of the process cannot be released for reuse until everything regarding the process has been released.
A process executing dirty native code in a NIF when it receives an exit signal will be set into an exiting state even if it is still executing dirty native code. Directly visible Erlang resources will be released, but the runtime system cannot force the native code to stop executing. The runtime system tries to prevent the execution of the dirty native code from effecting other processes by, for example, disabling functionality such as enif_send()
when used from a terminated process, but if the NIF is not well behaved it can still effect other processes. A well behaved dirty NIF should test if the process it is executing in has exited
, and if so stop executing.
In the general case, the heap of a process cannot be removed before all signals that it needs to send have been sent. Resources held by heap data are the memory blocks containing the heap, but also include things referred to from the heap such as off heap binaries, and resources held via NIF resource objects
on the heap.
Delivery of Signals
The amount of time that passes between the time a signal is sent and the arrival of the signal at the destination is unspecified but positive. If the receiver has terminated, the signal does not arrive, but it can trigger another signal. For example, a link
signal sent to a non-existing process triggers an exit
signal, which is sent back to where the link
signal originated from. When communicating over the distribution, signals can be lost if the distribution channel goes down.
The only signal ordering guarantee given is the following: if an entity sends multiple signals to the same destination entity, the order is preserved; that is, if A
sends a signal S1
to B
, and later sends signal S2
to B
, S1
is guaranteed not to arrive after S2
. Note that S1
may, or may not have been lost.
Irregularities
- Synchronous Error Checking
-
Some functionality that send signals have synchronous error checking when sending locally on a node and fail if the receiver is not present at the time when the signal is sent:
- The
send operator !
,erlang:send/2,3
, BIFs anderlang:send_nosuspend/2,3
BIFs when the receiver is identified by a name that is expected to be registered locally. -
erlang:link/1
-
erlang:group_leader/2
- The
- Unexpected Behaviours of Exit Signals
-
When a process sends an exit signal with exit reason
normal
to itself by callingerlang:exit(self(), normal)
it will be terminatedwhen the exit signal is received
. In all other cases when an exit signal with exit reasonnormal
is received, it is dropped.When an
exit signal with exit reason kill is received
, the action taken is different depending on whether the signal was sent due to a linked process terminating, or the signal was explicitly sent using theexit/2
BIF. When sent using theexit/2
BIF, the signal cannot betrapped
, while it can be trapped if the signal was sent due to a link. - Blocking Signaling Over Distribution
-
When sending a signal over a distribution channel, the sending process may be suspended even though the signal is supposed to be sent asynchronously. This is due to the built in flow control over the channel that has been present more or less for ever. When the size of the output buffer for the channel reach the distribution buffer busy limit, processes sending on the channel will be suspended until the size of the buffer shrinks below the limit. The size of the limit can be inspected by calling
erlang:system_info(dist_buf_busy_limit)
. Since this functionality has been present for so long, it is not possible to remove it, but it is possible to increase the limit to a point where it more or less never is reached using theerl
command line argument+zdbbl
. Note that if you do raise the limit like this, you need to take care of flow control yourself to ensure that you do not get into a situation with excessive memory usage.
The irregularities mentioned above cannot be fixed as they have been part of Erlang too long and it would break a lot of existing code.
12.7 Links
Two processes can be linked to each other. Also a process and a port that reside on the same node can be linked to each other. A link between two processes can be created if one of them calls the link/1
BIF with the process identifier of the other process as argument. Links can also be created using one the following spawn BIFs spawn_link()
, spawn_opt()
, or spawn_request()
. The spawn operation and the link operation will be performed atomically, in these cases.
If one of the participants of a link terminates, it will send an exit signal
to the other participant. The exit signal will contain the exit reason
of the terminated participant.
A link can be removed by calling the unlink/1
BIF.
Links are bidirectional and there can only be one link between two processes. Repeated calls to link()
have no effect. Either one of the involved processes may create or remove a link.
Links are used to monitor the behavior of other processes, see Error Handling
.
12.8 Error Handling
Erlang has a built-in feature for error handling between processes. Terminating processes emit exit signals to all linked processes, which can terminate as well or handle the exit in some way. This feature can be used to build hierarchical program structures where some processes are supervising other processes, for example, restarting them if they terminate abnormally.
See OTP Design Principles
for more information about OTP supervision trees, which use this feature.
Sending Exit Signals
When a process or port terminates
it will send exit signals to all processes and ports that it is linked
to. The exit signal will contain the following information:
- Sender identifier
The process or port identifier of the process or port that terminated.
- Receiver identifier
The process or port identifier of the process or port which the exit signal is sent to.
- The
link
flag This flag will be set indicating that the exit signal was sent due to a link.
- Exit reason
-
The exit reason of the process or port that terminated or the atom:
noproc
in case no process or port was found when setting up a link in a preceeding call to thelink(PidOrPort)
BIF. The process or port identified as sender of the exit signal will equal thePidOrPort
argument passed tolink/1
.noconnection
in case the linked processes resides on different nodes and the connection between the nodes was lost or could not be established. The process or port identified as sender of the exit signal might in this case still be alive.
Exit signals can also be sent explicitly by calling the exit(PidOrPort, Reason)
BIF. The exit signal is sent to the process or port identified by the PidOrPort
argument. The exit signal sent will contain the following information:
- Sender identifier
The process identifier of the process that called
exit/2
.- Receiver identifier
The process or port identifier of the process or port which the exit signal is sent to.
- The
link
flag This flag will not be set, indicating that this exit signal was not sent due to a link.
- Exit reason
The term passed as
Reason
in the call toexit/2
. IfReason
is the atomkill
, the receiver cannottrap the exit
signal and will unconditionally terminate when it receives the signal.
Receiving Exit Signals
What happens when a process receives an exit signal depends on:
The
trap exit
state of the receiver at the time when the exit signal is received.The exit reason of the exit signal.
The sender of the exit signal.
The state of the
link
flag of the exit signal. If thelink
flag is set, the exit signal was sent due to a link; otherwise, the exit signal was sent by a call to theexit/2
BIF.If the
link
flag is set, what happens also depends on whether thelink is still active or not
when the exit signal is received.
Based on the above states, the following will happen when an exit signal is received by a process:
-
The exit signal is silently dropped if:
the
link
flag of the exit signal is set and the corresponding link has been deactivated.the exit reason of the exit signal is the atom
normal
, the receiver is not trapping exits, and the receiver and sender are not the same process.
-
The receiving process is terminated if:
the
link
flag of the exit signal is not set, and the exit reason of the exit signal is the atomkill
. The receiving process will terminate with the atomkilled
as exit reason.the receiver is not trapping exits, and the exit reason is something other than the atom
normal
. Also, if thelink
flag of the exit signal is set, the link also needs to be active otherwise the exit signal will be dropped. The exit reason of the receiving process will equal the exit reason of the exit signal. Note that if thelink
flag is set, an exit reason ofkill
will not be converted tokilled
.the exit reason of the exit signal is the atom
normal
and the sender of the exit signal is the same process as the receiver. Thelink
flag cannot be set in this case. The exit reason of the receiving process will be the atomnormal
.
-
The exit signal is converted to a message signal and added to the end of the message queue of the receiver, if the receiver is trapping exits, the
link
flag of the exit signal is:not set, and the exit reason of the signal is not the atom
kill
.set, and the corresponding link is active. Note that an exit reason of
kill
will not terminate the process in this case and it will not be converted tokilled
.
The converted message will be on the form
{'EXIT', SenderID, Reason}
whereReason
equals the exit reason of the exit signal andSenderID
is the identifier of the process or port that sent the exit signal.
12.9 Monitors
An alternative to links are monitors. A process Pid1
can create a monitor for Pid2
by calling the BIF erlang:monitor(process, Pid2)
. The function returns a reference Ref
.
If Pid2
terminates with exit reason Reason
, a 'DOWN' message is sent to Pid1
:
{'DOWN', Ref, process, Pid2, Reason}
If Pid2
does not exist, the 'DOWN' message is sent immediately with Reason
set to noproc
.
Monitors are unidirectional. Repeated calls to erlang:monitor(process, Pid)
creates several independent monitors, and each one sends a 'DOWN' message when Pid
terminates.
A monitor can be removed by calling erlang:demonitor(Ref)
.
Monitors can be created for processes with registered names, also at other nodes.
12.10 Process Dictionary
Each process has its own process dictionary, accessed by calling the following BIFs:
put(Key, Value) get(Key) get() get_keys(Value) erase(Key) erase()
© 2010–2021 Ericsson AB
Licensed under the Apache License, Version 2.0.