diff --git a/ebin/egs.app b/ebin/egs.app index d68775a..a416580 100644 --- a/ebin/egs.app +++ b/ebin/egs.app @@ -6,6 +6,7 @@ egs, egs_app, egs_sup, + egs_exit_mon, reloader, egs_cron, egs_db, diff --git a/src/egs_exit_mon.erl b/src/egs_exit_mon.erl new file mode 100644 index 0000000..d0dbb54 --- /dev/null +++ b/src/egs_exit_mon.erl @@ -0,0 +1,47 @@ +%% @author Loïc Hoguin +%% @copyright 2010 Loïc Hoguin. +%% @doc General purpose module for monitoring exit signals of linked processes. +%% +%% This file is part of EGS. +%% +%% EGS is free software: you can redistribute it and/or modify +%% it under the terms of the GNU General Public License as published by +%% the Free Software Foundation, either version 3 of the License, or +%% (at your option) any later version. +%% +%% EGS is distributed in the hope that it will be useful, +%% but WITHOUT ANY WARRANTY; without even the implied warranty of +%% MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +%% GNU General Public License for more details. +%% +%% You should have received a copy of the GNU General Public License +%% along with EGS. If not, see . + +-module(egs_exit_mon). +-export([start_link/1]). %% External. +-export([start/1, loop/1]). %% Internal. + +%% @spec start_link(CleanupFn) -> {ok,Pid::pid()} +%% @doc Start the monitor and return the process' Pid. +start_link(CallbackFn) -> + Pid = spawn(?MODULE, start, [CallbackFn]), + {ok, Pid}. + +%% @spec start(CallbackFn) -> ok +%% @doc Start the main loop. +start(CallbackFn) -> + process_flag(trap_exit, true), + ?MODULE:loop(CallbackFn). + +%% @spec loop(CallbackFn) -> ok +%% @doc Main loop, trap exit messages and call the callback function. +loop(CallbackFn = {Module, Function}) -> + receive + {'EXIT', Pid, _} -> + spawn(Module, Function, [Pid]); + _ -> + reload + after 5000 -> + reload + end, + ?MODULE:loop(CallbackFn). diff --git a/src/psu/psu_game.erl b/src/psu/psu_game.erl index 49dd81b..4d952d6 100644 --- a/src/psu/psu_game.erl +++ b/src/psu/psu_game.erl @@ -18,8 +18,8 @@ %% along with EGS. If not, see . -module(psu_game). --export([start_link/1]). %% External. --export([supervisor_init/0, supervisor/0, listen/2, accept/2, process_init/2, process/0, char_select/0, area_load/4, loop/1]). %% Internal. +-export([start_link/1, cleanup/1]). %% External. +-export([listen/2, accept/2, process_init/2, process/0, char_select/0, area_load/4, loop/1]). %% Internal. -include("include/records.hrl"). -include("include/maps.hrl"). @@ -29,64 +29,41 @@ %% @spec start_link(Port) -> {ok,Pid::pid()} %% @doc Start the game server. start_link(Port) -> - SPid = spawn(?MODULE, supervisor_init, []), - LPid = spawn(?MODULE, listen, [Port, SPid]), + {ok, MPid} = egs_exit_mon:start_link({?MODULE, cleanup}), + LPid = spawn(?MODULE, listen, [Port, MPid]), {ok, LPid}. -%% @doc Game processes supervisor initialization. - -supervisor_init() -> - process_flag(trap_exit, true), - supervisor(). - -%% @doc Game processes supervisor. Make sure everything is cleaned up when an unexpected error occurs. - -supervisor() -> - receive - {'EXIT', Pid, _} -> - supervisor_close(Pid); - _ -> - reload - after 5000 -> - reload - end, - ?MODULE:supervisor(). - -%% @doc Close the connection for the given user and cleanup. - -supervisor_close(Pid) -> - try - User = egs_db:users_select_by_pid(Pid), - egs_db:users_delete(User#users.gid), - lists:foreach(fun(Other) -> Other#users.pid ! {psu_player_unspawn, User} end, egs_db:users_select_others_in_area(User)), - io:format("game (~p): quit~n", [User#users.gid]) - catch _:_ -> - ignore - end. +%% @spec cleanup(Pid) -> ok +%% @doc Cleanup the data associated with the failing process. +cleanup(Pid) -> + User = egs_db:users_select_by_pid(Pid), + egs_db:users_delete(User#users.gid), + lists:foreach(fun(Other) -> Other#users.pid ! {psu_player_unspawn, User} end, egs_db:users_select_others_in_area(User)), + io:format("game (~p): quit~n", [User#users.gid]). %% @doc Listen for connections. -listen(Port, SPid) -> +listen(Port, MPid) -> {ok, LSocket} = ssl:listen(Port, ?OPTIONS), - ?MODULE:accept(LSocket, SPid). + ?MODULE:accept(LSocket, MPid). %% @doc Accept connections. -accept(LSocket, SPid) -> +accept(LSocket, MPid) -> case ssl:transport_accept(LSocket, 5000) of {ok, CSocket} -> ssl:ssl_accept(CSocket), - Pid = spawn(?MODULE, process_init, [CSocket, SPid]), + Pid = spawn(?MODULE, process_init, [CSocket, MPid]), ssl:controlling_process(CSocket, Pid); _ -> reload end, - ?MODULE:accept(LSocket, SPid). + ?MODULE:accept(LSocket, MPid). %% @doc Initialize the client process by saving the socket to the process dictionary. -process_init(CSocket, SPid) -> - link(SPid), +process_init(CSocket, MPid) -> + link(MPid), put(socket, CSocket), send_0202(), process().