diff options
Diffstat (limited to 'lib/stdlib/src')
| -rw-r--r-- | lib/stdlib/src/pool.erl | 3 | ||||
| -rw-r--r-- | lib/stdlib/src/supervisor.erl | 86 | 
2 files changed, 63 insertions, 26 deletions
| diff --git a/lib/stdlib/src/pool.erl b/lib/stdlib/src/pool.erl index 7f5f23e26d..a3c9927ee9 100644 --- a/lib/stdlib/src/pool.erl +++ b/lib/stdlib/src/pool.erl @@ -95,6 +95,9 @@ pspawn_link(M, F, A) ->  start_nodes([], _, _) -> [];  start_nodes([Host|Tail], Name, Args) ->       case slave:start(Host, Name, Args) of  +	{error, {already_running, Node}} -> +	    io:format("Can't start node on host ~w due to ~w~n",[Host, {already_running, Node}]), +	    [Node | start_nodes(Tail, Name, Args)];  	{error, R} ->  	    io:format("Can't start node on host ~w due to ~w~n",[Host, R]),  	    start_nodes(Tail, Name, Args); diff --git a/lib/stdlib/src/supervisor.erl b/lib/stdlib/src/supervisor.erl index 368dc2e3e5..09a01a9aea 100644 --- a/lib/stdlib/src/supervisor.erl +++ b/lib/stdlib/src/supervisor.erl @@ -65,11 +65,13 @@  -type child() :: #child{}.  -define(DICT, dict). +-define(SETS, sets). +-define(SET, set).  -record(state, {name,  		strategy               :: strategy(),  		children = []          :: [child()], -		dynamics               :: ?DICT() | list(), +		dynamics               :: ?DICT() | ?SET(),  		intensity              :: non_neg_integer(),  		period                 :: pos_integer(),  		restarts = [], @@ -138,7 +140,7 @@ delete_child(Supervisor, Name) ->  %%-----------------------------------------------------------------  -type term_err() :: 'not_found' | 'simple_one_for_one'. --spec terminate_child(sup_ref(), term()) -> 'ok' | {'error', term_err()}. +-spec terminate_child(sup_ref(), pid() | term()) -> 'ok' | {'error', term_err()}.  terminate_child(Supervisor, Name) ->      call(Supervisor, {terminate_child, Name}). @@ -297,8 +299,26 @@ handle_call({start_child, EArgs}, _From, State) when ?is_simple(State) ->  	    {reply, What, State}      end; -%%% The requests terminate_child, delete_child and restart_child are  -%%% invalid for simple_one_for_one supervisors. +%% terminate_child for simple_one_for_one can only be done with pid +handle_call({terminate_child, Name}, _From, State) when not is_pid(Name), +							?is_simple(State) -> +    {reply, {error, simple_one_for_one}, State}; + +handle_call({terminate_child, Name}, _From, State) -> +    case get_child(Name, State, ?is_simple(State)) of +	{value, Child} -> +	    case do_terminate(Child, State#state.name) of +		#child{restart_type=RT} when RT=:=temporary; ?is_simple(State) -> +		    {reply, ok, state_del_child(Child, State)}; +		NChild -> +		    {reply, ok, replace_child(NChild, State)} +		end; +	false -> +	    {reply, {error, not_found}, State} +    end; + +%%% The requests delete_child and restart_child are invalid for +%%% simple_one_for_one supervisors.  handle_call({_Req, _Data}, _From, State) when ?is_simple(State) ->      {reply, {error, simple_one_for_one}, State}; @@ -341,25 +361,12 @@ handle_call({delete_child, Name}, _From, State) ->  	    {reply, {error, not_found}, State}      end; -handle_call({terminate_child, Name}, _From, State) -> -    case get_child(Name, State) of -	{value, Child} -> -	    case do_terminate(Child, State#state.name) of -		#child{restart_type = temporary} = NChild -> -		    {reply, ok, state_del_child(NChild, State)}; -		NChild -> -		    {reply, ok, replace_child(NChild, State)} -		end; -	_ -> -	    {reply, {error, not_found}, State} -    end; -  handle_call(which_children, _From, #state{children = [#child{restart_type = temporary,  							     child_type = CT,  							     modules = Mods}]} =  		State) when ?is_simple(State) -> -    Reply = lists:map(fun(Pid) -> {undefined, Pid, CT, Mods} end, dynamics_db(temporary, -									      State#state.dynamics)), +    Reply = lists:map(fun(Pid) -> {undefined, Pid, CT, Mods} end, +                      ?SETS:to_list(dynamics_db(temporary, State#state.dynamics))),      {reply, Reply, State};  handle_call(which_children, _From, #state{children = [#child{restart_type = RType, @@ -384,7 +391,7 @@ handle_call(count_children, _From, #state{children = [#child{restart_type = temp  							     child_type = CT}]} = State)    when ?is_simple(State) ->      {Active, Count} = -	lists:foldl(fun(Pid, {Alive, Tot}) -> +	?SETS:fold(fun(Pid, {Alive, Tot}) ->  			   if is_pid(Pid) -> {Alive+1, Tot +1};  			      true -> {Alive, Tot + 1} end  		   end, {0, 0}, dynamics_db(temporary, State#state.dynamics)), @@ -795,24 +802,27 @@ save_child(Child, #state{children = Children} = State) ->      State#state{children = [Child |Children]}.  save_dynamic_child(temporary, Pid, _, #state{dynamics = Dynamics} = State) -> -    State#state{dynamics = [Pid | dynamics_db(temporary, Dynamics)]}; +    State#state{dynamics = ?SETS:add_element(Pid, dynamics_db(temporary, Dynamics))};  save_dynamic_child(RestartType, Pid, Args, #state{dynamics = Dynamics} = State) ->      State#state{dynamics = ?DICT:store(Pid, Args, dynamics_db(RestartType, Dynamics))}.  dynamics_db(temporary, undefined) -> -    []; +    ?SETS:new();  dynamics_db(_, undefined) ->      ?DICT:new();  dynamics_db(_,Dynamics) ->      Dynamics. -dynamic_child_args(_, Dynamics) when is_list(Dynamics)-> -    {ok, undefined};  dynamic_child_args(Pid, Dynamics) -> -    ?DICT:find(Pid, Dynamics). +    case ?SETS:is_set(Dynamics) of +        true -> +            {ok, undefined}; +        false -> +            ?DICT:find(Pid, Dynamics) +    end.  state_del_child(#child{pid = Pid, restart_type = temporary}, State) when ?is_simple(State) -> -    NDynamics = lists:delete(Pid, dynamics_db(temporary, State#state.dynamics)), +    NDynamics = ?SETS:del_element(Pid, dynamics_db(temporary, State#state.dynamics)),      State#state{dynamics = NDynamics};  state_del_child(#child{pid = Pid, restart_type = RType}, State) when ?is_simple(State) ->      NDynamics = ?DICT:erase(Pid, dynamics_db(RType, State#state.dynamics)), @@ -849,7 +859,31 @@ split_child(_, [], After) ->      {lists:reverse(After), []}.  get_child(Name, State) -> +    get_child(Name, State, false). +get_child(Pid, State, AllowPid) when AllowPid, is_pid(Pid) -> +    get_dynamic_child(Pid, State); +get_child(Name, State, _) ->      lists:keysearch(Name, #child.name, State#state.children). + +get_dynamic_child(Pid, #state{children=[Child], dynamics=Dynamics}) -> +    case is_dynamic_pid(Pid, dynamics_db(Child#child.restart_type, Dynamics)) of +	true -> +	    {value, Child#child{pid=Pid}}; +	false -> +	    case erlang:is_process_alive(Pid) of +		true -> false; +		false -> {value, Child} +	    end +    end. + +is_dynamic_pid(Pid, Dynamics) -> +    case ?SETS:is_set(Dynamics) of +	true -> +	    ?SETS:is_element(Pid, Dynamics); +	false -> +	    ?DICT:is_key(Pid, Dynamics) +    end. +  replace_child(Child, State) ->      Chs = do_replace_child(Child, State#state.children),      State#state{children = Chs}. | 
