2 -behaviour(gen_server).
6 -export([start_link/1]).
18 -record(state, {id :: integer()
21 ,neighbors :: list(atom())
22 ,live_neighbors :: integer()
23 ,num_neighbors :: integer()
24 ,replies_pending :: integer()
28 %% ============================================================================
30 %% ============================================================================
32 start_link({_ID, Name, _NeighborNames}=Datum) ->
33 ServerName = {local, Name},
36 gen_server:start_link(ServerName, ?MODULE, Args, Opts).
39 %% ============================================================================
41 %% ============================================================================
43 init([{ID, Name, NeighborNames}]) ->
46 ,cell_state=crypto:rand_uniform(0, 2)
47 ,neighbors=NeighborNames
48 ,num_neighbors=length(NeighborNames)
55 terminate(_Reason, State) ->
59 code_change(_Old, State, _Other) ->
63 handle_call(_Msg, _From, State) ->
67 handle_cast(_Msg, State) ->
71 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
76 ,num_neighbors=NumNeighbors
79 ok = send_all(Neighbors, {request_state, Name}),
80 {noreply, State#state{replies_pending=NumNeighbors}};
83 handle_info({request_state, Requester}, State) ->
84 Requester ! {response_state, State#state.cell_state},
88 handle_info({response_state, NeighborState},
90 ,replies_pending=Pending
92 ,live_neighbors=LiveNeighbors
95 NewPending = Pending - 1,
96 NewLiveNeighbors = LiveNeighbors + NeighborState,
98 NewState = State#state{replies_pending=NewPending
99 ,live_neighbors=NewLiveNeighbors
104 NewCellState = new_state(CellState, NewLiveNeighbors),
105 ok = time:cast({tock, {ID, NewCellState}}),
107 {noreply, NewState#state{live_neighbors=0
108 ,cell_state=NewCellState
115 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
118 handle_info(_Msg, State) ->
122 %% ============================================================================
124 %% ============================================================================
126 send_all([], _) -> ok;
127 send_all([PID | PIDs], Msg) ->
132 new_state(1, LiveNeighbors) when LiveNeighbors < 2 -> 0;
133 new_state(1, LiveNeighbors) when LiveNeighbors < 4 -> 1;
134 new_state(1, LiveNeighbors) when LiveNeighbors > 3 -> 0;
135 new_state(0, LiveNeighbors) when LiveNeighbors =:= 3 -> 1;
136 new_state(State, _LiveNeighbors) -> State.