game: Pattern match the command header only once.
This commit is contained in:
parent
0b8ecb3971
commit
9d130db310
102
src/egs_game.erl
102
src/egs_game.erl
@ -96,8 +96,8 @@ accept(LSocket, SPid) ->
|
|||||||
process(CSocket, Version) ->
|
process(CSocket, Version) ->
|
||||||
case egs_proto:packet_recv(CSocket, 5000) of
|
case egs_proto:packet_recv(CSocket, 5000) of
|
||||||
{ok, Orig} ->
|
{ok, Orig} ->
|
||||||
<< _:32, Command:16/unsigned-integer, _/bits >> = Orig,
|
{command, Command, _, Data} = egs_proto:packet_parse(Orig),
|
||||||
process_handle(Command, CSocket, Version, Orig);
|
process_handle(Command, CSocket, Version, Data);
|
||||||
{error, timeout} ->
|
{error, timeout} ->
|
||||||
reload,
|
reload,
|
||||||
?MODULE:process(CSocket, Version);
|
?MODULE:process(CSocket, Version);
|
||||||
@ -107,8 +107,7 @@ process(CSocket, Version) ->
|
|||||||
|
|
||||||
%% @doc Game server auth request handler.
|
%% @doc Game server auth request handler.
|
||||||
|
|
||||||
process_handle(16#020d, CSocket, Version, Orig) ->
|
process_handle(16#020d, CSocket, Version, << GID:32/little-unsigned-integer, Auth:32/bits, _/bits >>) ->
|
||||||
<< _:352, GID:32/little-unsigned-integer, Auth:32/bits, _/bits >> = Orig,
|
|
||||||
case egs_db:users_select(GID) of
|
case egs_db:users_select(GID) of
|
||||||
error ->
|
error ->
|
||||||
log(GID, "can't find user, closing"),
|
log(GID, "can't find user, closing"),
|
||||||
@ -131,8 +130,7 @@ process_handle(16#020d, CSocket, Version, Orig) ->
|
|||||||
|
|
||||||
%% @doc Platform information handler. Obtain the game version.
|
%% @doc Platform information handler. Obtain the game version.
|
||||||
|
|
||||||
process_handle(16#080e, CSocket, _, Orig) ->
|
process_handle(16#080e, CSocket, _, << _:64, Version:32/little-unsigned-integer, _/bits >>) ->
|
||||||
<< _:416, Version:32/little-unsigned-integer, _/bits >> = Orig,
|
|
||||||
?MODULE:process(CSocket, Version);
|
?MODULE:process(CSocket, Version);
|
||||||
|
|
||||||
%% @doc Unknown command handler. Do nothing.
|
%% @doc Unknown command handler. Do nothing.
|
||||||
@ -147,8 +145,8 @@ process_handle(Command, CSocket, Version, _) ->
|
|||||||
char_select(CSocket, GID, Version) ->
|
char_select(CSocket, GID, Version) ->
|
||||||
case egs_proto:packet_recv(CSocket, 5000) of
|
case egs_proto:packet_recv(CSocket, 5000) of
|
||||||
{ok, Orig} ->
|
{ok, Orig} ->
|
||||||
<< _:32, Command:16/unsigned-integer, _/bits >> = Orig,
|
{command, Command, _, Data} = egs_proto:packet_parse(Orig),
|
||||||
char_select_handle(Command, CSocket, GID, Version, Orig);
|
char_select_handle(Command, CSocket, GID, Version, Data);
|
||||||
{error, timeout} ->
|
{error, timeout} ->
|
||||||
egs_proto:send_keepalive(CSocket),
|
egs_proto:send_keepalive(CSocket),
|
||||||
reload,
|
reload,
|
||||||
@ -160,17 +158,15 @@ char_select(CSocket, GID, Version) ->
|
|||||||
|
|
||||||
%% @doc Character selection handler.
|
%% @doc Character selection handler.
|
||||||
|
|
||||||
char_select_handle(16#020b, CSocket, GID, Version, Orig) ->
|
char_select_handle(16#020b, CSocket, GID, Version, << Number:32/little-unsigned-integer, _/bits >>) ->
|
||||||
<< _:352, Number:32/little-unsigned-integer, _/bits >> = Orig,
|
|
||||||
log(GID, "selected character ~b", [Number]),
|
log(GID, "selected character ~b", [Number]),
|
||||||
char_select_load(CSocket, GID, Version, Number);
|
char_select_load(CSocket, GID, Version, Number);
|
||||||
|
|
||||||
%% @doc Character creation handler.
|
%% @doc Character creation handler.
|
||||||
|
|
||||||
char_select_handle(16#0d02, CSocket, GID, Version, Orig) ->
|
char_select_handle(16#0d02, CSocket, GID, Version, << Number:32/little-unsigned-integer, Char/bits >>) ->
|
||||||
log(GID, "character creation"),
|
log(GID, "character creation"),
|
||||||
User = egs_db:users_select(GID),
|
User = egs_db:users_select(GID),
|
||||||
<< _:352, Number:32/little-unsigned-integer, Char/bits >> = Orig,
|
|
||||||
_ = file:make_dir(io_lib:format("save/~s", [User#users.folder])),
|
_ = file:make_dir(io_lib:format("save/~s", [User#users.folder])),
|
||||||
file:write_file(io_lib:format("save/~s/~b-character", [User#users.folder, Number]), Char),
|
file:write_file(io_lib:format("save/~s/~b-character", [User#users.folder, Number]), Char),
|
||||||
file:write_file(io_lib:format("save/~s/~b-character.options", [User#users.folder, Number]), << 0:192 >>),
|
file:write_file(io_lib:format("save/~s/~b-character.options", [User#users.folder, Number]), << 0:192 >>),
|
||||||
@ -480,17 +476,13 @@ loop(CSocket, GID, Version, SoFar) ->
|
|||||||
end.
|
end.
|
||||||
|
|
||||||
%% @doc Dispatch the command to the right handler.
|
%% @doc Dispatch the command to the right handler.
|
||||||
%% Command 0b05 uses the channel for something else. Conflicts could occur. Better to just ignore it anyway.
|
|
||||||
|
|
||||||
dispatch(CSocket, GID, Version, Orig) ->
|
dispatch(CSocket, GID, Version, Orig) ->
|
||||||
<< _:32, Command:16/unsigned-integer, Channel:8/little-unsigned-integer, _/bits >> = Orig,
|
{command, Command, Channel, Data} = egs_proto:packet_parse(Orig),
|
||||||
case [Command, Channel] of
|
case Channel of
|
||||||
[16#0b05, _] ->
|
ignore -> ignore;
|
||||||
ignore;
|
1 -> broadcast(Command, GID, Orig);
|
||||||
[_, 1] ->
|
_ -> handle(Command, CSocket, GID, Version, Data)
|
||||||
broadcast(Command, GID, Orig);
|
|
||||||
_ ->
|
|
||||||
handle(Command, CSocket, GID, Version, Orig)
|
|
||||||
end.
|
end.
|
||||||
|
|
||||||
%% @doc Position change broadcast handler. Save the position and then dispatch it.
|
%% @doc Position change broadcast handler. Save the position and then dispatch it.
|
||||||
@ -551,8 +543,8 @@ handle(16#0102, _, _, _, _) ->
|
|||||||
%% @todo Apparently B is always ItemID+1. Not sure why.
|
%% @todo Apparently B is always ItemID+1. Not sure why.
|
||||||
%% @todo Currently use a separate file for the data sent for the weapons.
|
%% @todo Currently use a separate file for the data sent for the weapons.
|
||||||
|
|
||||||
handle(16#0105, CSocket, GID, _, Orig) ->
|
handle(16#0105, CSocket, GID, _, Data) ->
|
||||||
<< _:384, A:32/little-unsigned-integer, ItemID:8, Action:8, _:8, B:8, C:32/little-unsigned-integer, _/bits >> = Orig,
|
<< _:32, A:32/little-unsigned-integer, ItemID:8, Action:8, _:8, B:8, C:32/little-unsigned-integer, _/bits >> = Data,
|
||||||
log(GID, "0105 action ~b item ~b (~b ~b ~b)", [Action, ItemID, A, B, C]),
|
log(GID, "0105 action ~b item ~b (~b ~b ~b)", [Action, ItemID, A, B, C]),
|
||||||
Category = case ItemID of
|
Category = case ItemID of
|
||||||
% units would be 8, traps would be 12
|
% units would be 8, traps would be 12
|
||||||
@ -601,8 +593,8 @@ handle(16#0105, CSocket, GID, _, Orig) ->
|
|||||||
%% @doc Shop listing request. Currently return the normal item shop for everything.
|
%% @doc Shop listing request. Currently return the normal item shop for everything.
|
||||||
%% @todo Return the other shops appropriately.
|
%% @todo Return the other shops appropriately.
|
||||||
|
|
||||||
handle(16#010a, CSocket, GID, _, Orig) ->
|
handle(16#010a, CSocket, GID, _, Data) ->
|
||||||
<< _:384, A:32/little-unsigned-integer, B:32/little-unsigned-integer, C:32/little-unsigned-integer >> = Orig,
|
<< _:32, A:32/little-unsigned-integer, B:32/little-unsigned-integer, C:32/little-unsigned-integer >> = Data,
|
||||||
log(GID, "shop listing request (~b, ~b, ~b)", [A, B, C]),
|
log(GID, "shop listing request (~b, ~b, ~b)", [A, B, C]),
|
||||||
{ok, File} = file:read_file("p/itemshop.bin"),
|
{ok, File} = file:read_file("p/itemshop.bin"),
|
||||||
Packet = << 16#010a0300:32, 0:64, GID:32/little-unsigned-integer, 0:64, 16#00011300:32,
|
Packet = << 16#010a0300:32, 0:64, GID:32/little-unsigned-integer, 0:64, 16#00011300:32,
|
||||||
@ -612,8 +604,8 @@ handle(16#010a, CSocket, GID, _, Orig) ->
|
|||||||
%% @doc Character death, and more, handler. Warp to 4th floor for now.
|
%% @doc Character death, and more, handler. Warp to 4th floor for now.
|
||||||
%% @todo Recover from death correctly.
|
%% @todo Recover from death correctly.
|
||||||
|
|
||||||
handle(16#0110, CSocket, GID, _, Orig) ->
|
handle(16#0110, CSocket, GID, _, Data) ->
|
||||||
<< _:384, A:32/little-unsigned-integer, B:32/little-unsigned-integer, C:32/little-unsigned-integer >> = Orig,
|
<< _:32, A:32/little-unsigned-integer, B:32/little-unsigned-integer, C:32/little-unsigned-integer >> = Data,
|
||||||
case B of
|
case B of
|
||||||
2 -> % triggered when looking at the type menu
|
2 -> % triggered when looking at the type menu
|
||||||
send_0113(CSocket, GID);
|
send_0113(CSocket, GID);
|
||||||
@ -637,8 +629,7 @@ handle(16#021d, CSocket, _, _, _) ->
|
|||||||
%% When selecting 'Reload', reload the character in the current lobby.
|
%% When selecting 'Reload', reload the character in the current lobby.
|
||||||
%% @todo There's probably an entryid given in the uni selection packet.
|
%% @todo There's probably an entryid given in the uni selection packet.
|
||||||
|
|
||||||
handle(16#021f, CSocket, GID, _, Orig) ->
|
handle(16#021f, CSocket, GID, _, << Uni:32/little-unsigned-integer, _/bits >>) ->
|
||||||
<< _:352, Uni:32/little-unsigned-integer, _/bits >> = Orig,
|
|
||||||
case Uni of
|
case Uni of
|
||||||
0 -> % cancelled uni selection
|
0 -> % cancelled uni selection
|
||||||
ignore;
|
ignore;
|
||||||
@ -669,13 +660,13 @@ handle(16#0302, _, GID, _, _) ->
|
|||||||
%% Disregard the name sent by the server in later versions of the game. Use the name saved in memory instead, to prevent client-side editing.
|
%% Disregard the name sent by the server in later versions of the game. Use the name saved in memory instead, to prevent client-side editing.
|
||||||
%% @todo Only broadcast to people in the same map.
|
%% @todo Only broadcast to people in the same map.
|
||||||
|
|
||||||
handle(16#0304, _, GID, Version, Orig) ->
|
handle(16#0304, _, GID, Version, Data) ->
|
||||||
User = egs_db:users_select(GID),
|
User = egs_db:users_select(GID),
|
||||||
case Version of
|
case Version of
|
||||||
0 -> % AOTI v2.000
|
0 -> % AOTI v2.000
|
||||||
<< _:416, Modifiers:128/bits, Message/bits >> = Orig;
|
<< _:64, Modifiers:128/bits, Message/bits >> = Data;
|
||||||
_ -> % Above
|
_ -> % Above
|
||||||
<< _:416, Modifiers:128/bits, _:512, Message/bits >> = Orig
|
<< _:64, Modifiers:128/bits, _:512, Message/bits >> = Data
|
||||||
end,
|
end,
|
||||||
[LogName|_] = re:split(User#users.charname, "\\0\\0", [{return, binary}]),
|
[LogName|_] = re:split(User#users.charname, "\\0\\0", [{return, binary}]),
|
||||||
[TmpMessage|_] = re:split(Message, "\\0\\0", [{return, binary}]),
|
[TmpMessage|_] = re:split(Message, "\\0\\0", [{return, binary}]),
|
||||||
@ -686,8 +677,8 @@ handle(16#0304, _, GID, Version, Orig) ->
|
|||||||
%% @todo Handle this packet properly.
|
%% @todo Handle this packet properly.
|
||||||
%% @todo Spawn cleared response event shouldn't be handled following this packet but when we see the spawn actually dead HP-wise.
|
%% @todo Spawn cleared response event shouldn't be handled following this packet but when we see the spawn actually dead HP-wise.
|
||||||
|
|
||||||
handle(16#0402, CSocket, GID, _, Orig) ->
|
handle(16#0402, CSocket, GID, _, Data) ->
|
||||||
<< _:352, SpawnID:32/little-unsigned-integer, _:64, Type:32/little-unsigned-integer, _:64 >> = Orig,
|
<< SpawnID:32/little-unsigned-integer, _:64, Type:32/little-unsigned-integer, _:64 >> = Data,
|
||||||
case Type of
|
case Type of
|
||||||
7 -> % spawn cleared @todo 1201 sent back with same values apparently, but not always
|
7 -> % spawn cleared @todo 1201 sent back with same values apparently, but not always
|
||||||
if SpawnID =:= 70 ->
|
if SpawnID =:= 70 ->
|
||||||
@ -703,8 +694,8 @@ handle(16#0402, CSocket, GID, _, Orig) ->
|
|||||||
|
|
||||||
%% @todo Handle this packet.
|
%% @todo Handle this packet.
|
||||||
|
|
||||||
handle(16#0404, CSocket, GID, _, Orig) ->
|
handle(16#0404, CSocket, GID, _, Data) ->
|
||||||
<< _:352, A:32/little-unsigned-integer, B:32/little-unsigned-integer >> = Orig,
|
<< A:32/little-unsigned-integer, B:32/little-unsigned-integer >> = Data,
|
||||||
log(GID, "unknown command 0404: ~b ~b", [A, B]),
|
log(GID, "unknown command 0404: ~b ~b", [A, B]),
|
||||||
send_1205(CSocket, GID, A, B);
|
send_1205(CSocket, GID, A, B);
|
||||||
|
|
||||||
@ -712,17 +703,17 @@ handle(16#0404, CSocket, GID, _, Orig) ->
|
|||||||
%% Rooms are handled differently than normal lobbies.
|
%% Rooms are handled differently than normal lobbies.
|
||||||
%% @todo When changing lobby to the room, 0230 must also be sent. Same when going from room to lobby.
|
%% @todo When changing lobby to the room, 0230 must also be sent. Same when going from room to lobby.
|
||||||
|
|
||||||
handle(16#0807, CSocket, GID, _, Orig) ->
|
handle(16#0807, CSocket, GID, _, Data) ->
|
||||||
<< _:352, QuestID:32/little-unsigned-integer, ZoneID:16/little-unsigned-integer,
|
<< QuestID:32/little-unsigned-integer, ZoneID:16/little-unsigned-integer,
|
||||||
MapID:16/little-unsigned-integer, EntryID:16/little-unsigned-integer, _/bits >> = Orig,
|
MapID:16/little-unsigned-integer, EntryID:16/little-unsigned-integer, _/bits >> = Data,
|
||||||
log(GID, "map change (~b,~b,~b,~b)", [QuestID, ZoneID, MapID, EntryID]),
|
log(GID, "map change (~b,~b,~b,~b)", [QuestID, ZoneID, MapID, EntryID]),
|
||||||
area_load(CSocket, GID, QuestID, ZoneID, MapID, EntryID);
|
area_load(CSocket, GID, QuestID, ZoneID, MapID, EntryID);
|
||||||
|
|
||||||
%% @doc Mission counter handler.
|
%% @doc Mission counter handler.
|
||||||
|
|
||||||
handle(16#0811, CSocket, GID, _, Orig) ->
|
handle(16#0811, CSocket, GID, _, Data) ->
|
||||||
<< _:352, QuestID:32/little-unsigned-integer, ZoneID:16/little-unsigned-integer,
|
<< QuestID:32/little-unsigned-integer, ZoneID:16/little-unsigned-integer,
|
||||||
MapID:16/little-unsigned-integer, EntryID:16/little-unsigned-integer, _/bits >> = Orig,
|
MapID:16/little-unsigned-integer, EntryID:16/little-unsigned-integer, _/bits >> = Data,
|
||||||
log(GID, "mission counter (~b,~b,~b,~b)", [QuestID, ZoneID, MapID, EntryID]),
|
log(GID, "mission counter (~b,~b,~b,~b)", [QuestID, ZoneID, MapID, EntryID]),
|
||||||
counter_load(CSocket, GID, QuestID, ZoneID, MapID, EntryID);
|
counter_load(CSocket, GID, QuestID, ZoneID, MapID, EntryID);
|
||||||
|
|
||||||
@ -735,15 +726,13 @@ handle(16#0812, CSocket, GID, _, _) ->
|
|||||||
%% @doc Item description request.
|
%% @doc Item description request.
|
||||||
%% @todo Send something other than just "dammy".
|
%% @todo Send something other than just "dammy".
|
||||||
|
|
||||||
handle(16#0a10, CSocket, GID, _, Orig) ->
|
handle(16#0a10, CSocket, GID, _, << ItemID:32/unsigned-integer >>) ->
|
||||||
<< _:352, ItemID:32/unsigned-integer >> = Orig,
|
|
||||||
send_0a11(CSocket, GID, ItemID, "dammy");
|
send_0a11(CSocket, GID, ItemID, "dammy");
|
||||||
|
|
||||||
%% @doc Start mission handler.
|
%% @doc Start mission handler.
|
||||||
%% @todo Forward the mission start to other players of the same party, whatever their location is.
|
%% @todo Forward the mission start to other players of the same party, whatever their location is.
|
||||||
|
|
||||||
handle(16#0c01, CSocket, GID, _, Orig) ->
|
handle(16#0c01, CSocket, GID, _, << QuestID:32/little-unsigned-integer >>) ->
|
||||||
<< _:352, QuestID:32/little-unsigned-integer >> = Orig,
|
|
||||||
log(GID, "start mission ~b", [QuestID]),
|
log(GID, "start mission ~b", [QuestID]),
|
||||||
send_170c(CSocket, GID),
|
send_170c(CSocket, GID),
|
||||||
send_1020(CSocket, GID),
|
send_1020(CSocket, GID),
|
||||||
@ -786,17 +775,16 @@ handle(16#0c0f, CSocket, GID, _, _) ->
|
|||||||
%% Just reply with a success value for now.
|
%% Just reply with a success value for now.
|
||||||
%% @todo God save the flags.
|
%% @todo God save the flags.
|
||||||
|
|
||||||
handle(16#0d04, CSocket, GID, _, Orig) ->
|
handle(16#0d04, CSocket, GID, _, Data) ->
|
||||||
<< _:352, Flag:128/bits, A:16/bits, _:8, B/bits >> = Orig,
|
<< Flag:128/bits, A:16/bits, _:8, B/bits >> = Data,
|
||||||
log(GID, "flag handler for ~s", [re:replace(Flag, "\\0+", "", [global, {return, binary}])]),
|
log(GID, "flag handler for ~s", [re:replace(Flag, "\\0+", "", [global, {return, binary}])]),
|
||||||
Packet = << 16#0d040300:32, 0:160, 16#00011300:32, GID:32/little-unsigned-integer, 0:64, Flag/binary, A/binary, 1, B/binary >>,
|
Packet = << 16#0d040300:32, 0:160, 16#00011300:32, GID:32/little-unsigned-integer, 0:64, Flag/binary, A/binary, 1, B/binary >>,
|
||||||
egs_proto:packet_send(CSocket, Packet);
|
egs_proto:packet_send(CSocket, Packet);
|
||||||
|
|
||||||
%% @doc Options changes handler.
|
%% @doc Options changes handler.
|
||||||
|
|
||||||
handle(16#0d07, _, GID, _, Orig) ->
|
handle(16#0d07, _, GID, _, << Options/bits >>) ->
|
||||||
log(GID, "options changes"),
|
log(GID, "options changes"),
|
||||||
<< _:352, Options/bits >> = Orig,
|
|
||||||
User = egs_db:users_select(GID),
|
User = egs_db:users_select(GID),
|
||||||
file:write_file(io_lib:format("save/~s/~b-character.options", [User#users.folder, User#users.charnumber]), Options);
|
file:write_file(io_lib:format("save/~s/~b-character.options", [User#users.folder, User#users.charnumber]), Options);
|
||||||
|
|
||||||
@ -804,15 +792,15 @@ handle(16#0d07, _, GID, _, Orig) ->
|
|||||||
%% @todo Finish the work on it.
|
%% @todo Finish the work on it.
|
||||||
%% @todo First value at 2C is the number of hits. We don't need to know it though.
|
%% @todo First value at 2C is the number of hits. We don't need to know it though.
|
||||||
|
|
||||||
handle(16#0e00, CSocket, GID, _, Orig) ->
|
handle(16#0e00, CSocket, GID, _, Data) ->
|
||||||
<< _:448, Data/bits >> = Orig,
|
<< _:96, Hits/bits >> = Data,
|
||||||
handle_hits(CSocket, GID, Data);
|
handle_hits(CSocket, GID, Hits);
|
||||||
|
|
||||||
%% @doc Object event handler.
|
%% @doc Object event handler.
|
||||||
%% @todo Handle all events appropriately.
|
%% @todo Handle all events appropriately.
|
||||||
|
|
||||||
handle(16#0f0a, CSocket, GID, _, Orig) ->
|
handle(16#0f0a, CSocket, GID, _, Data) ->
|
||||||
<< _:448, A:32/little-unsigned-integer, _:64, B:32/little-unsigned-integer, _:272, Action:8, _/bits >> = Orig,
|
<< _:96, A:32/little-unsigned-integer, _:64, B:32/little-unsigned-integer, _:272, Action:8, _/bits >> = Data,
|
||||||
case Action of
|
case Action of
|
||||||
0 -> % warp
|
0 -> % warp
|
||||||
ignore;
|
ignore;
|
||||||
@ -883,8 +871,8 @@ handle(16#1710, CSocket, GID, _, _) ->
|
|||||||
%% @doc Dialog request handler. Do what we can.
|
%% @doc Dialog request handler. Do what we can.
|
||||||
%% @todo Handle correctly.
|
%% @todo Handle correctly.
|
||||||
|
|
||||||
handle(16#1a01, CSocket, GID, _, Orig) ->
|
handle(16#1a01, CSocket, GID, _, Data) ->
|
||||||
<< _:384, A:8, B:8, _:16, C:8, _/bits >> = Orig,
|
<< _:32, A:8, B:8, _:16, C:8, _/bits >> = Data,
|
||||||
case [A, B, C] of
|
case [A, B, C] of
|
||||||
[ 0, 0, 2] ->
|
[ 0, 0, 2] ->
|
||||||
log(GID, "lumilass (and more?)"),
|
log(GID, "lumilass (and more?)"),
|
||||||
|
@ -119,6 +119,16 @@ packet_split(Packet, Result) ->
|
|||||||
end
|
end
|
||||||
end.
|
end.
|
||||||
|
|
||||||
|
%% @doc Parse the packet header returns the header information along with the data chunk.
|
||||||
|
%% 0b05 is handled differently because it's only 16 bytes long and use a different format.
|
||||||
|
|
||||||
|
packet_parse(<< _:32, 16#0b05:16, _/bits >>) ->
|
||||||
|
{command, 16#0b05, ignore, ignore};
|
||||||
|
|
||||||
|
packet_parse(Orig) ->
|
||||||
|
<< _:32, Command:16/unsigned-integer, Channel:8, _:296, Data/bits >> = Orig,
|
||||||
|
{command, Command, Channel, Data}.
|
||||||
|
|
||||||
%% @doc Shortcut for send_global/4.
|
%% @doc Shortcut for send_global/4.
|
||||||
|
|
||||||
send_global(CSocket, Type, Message) ->
|
send_global(CSocket, Type, Message) ->
|
||||||
|
Loading…
Reference in New Issue
Block a user