psu_proto: Clean parsing separation. Handle events rather than packets.

Lou can now equip a wepon.
This commit is contained in:
Loïc Hoguin 2010-08-19 23:43:19 +02:00
parent f6305c3c76
commit 6dd159dc1e
2 changed files with 154 additions and 66 deletions

View File

@ -497,11 +497,16 @@ loop(SoFar) ->
%% @doc Dispatch the command to the right handler.
dispatch(Orig) ->
{command, Command, Channel, Data} = psu_proto:packet_parse(Orig),
case Channel of
ignore -> ignore;
1 -> broadcast(Command, Orig);
_ -> handle(Command, Data)
case psu_proto:parse(Orig) of
{command, Command, Channel, Data} ->
case Channel of
1 -> broadcast(Command, Orig);
_ -> handle(Command, Data)
end;
ignore ->
ignore;
Event ->
event(Event)
end.
%% @doc Position change broadcast handler. Save the position and then dispatch it.
@ -550,21 +555,15 @@ broadcast(Command, Orig)
lists:foreach(fun(User) -> User#egs_user_model.pid ! {psu_broadcast, Packet} end, SpawnList)
end.
%% @doc Movement (non-broadcast) handler. Do nothing.
handle(16#0102, _) ->
ignore;
%% @doc Weapon equip, unequip, item drop, and more... handler. Do what we can.
%% @todo A and B are unknown.
%% Melee uses a format similar to: AAAA--BBCCCC----DDDDDDDDEE----FF with
%% AAAA the attack sound effect, BB the range, CCCC and DDDDDDDD unknown but related to angular range or similar, EE number of targets and FF the model.
%% Bullets and tech weapons formats are unknown but likely use a slightly different format.
%% @todo Others probably want to see that you changed your weapon.
%% @todo Apparently B is always ItemID+1. Not sure why.
%% @todo Currently use a separate file for the data sent for the weapons.
%% @todo We must also handle here the NPC characters. PartyPos can be used for that, and more info in unknown values maybe too?
handle(16#0105, Data) ->
<< _:32, PartyPos:32/little-unsigned-integer, ItemID:8, Action:8, _:8, A:8, B:32/little-unsigned-integer, _/bits >> = Data,
log("0105 action ~b item ~b partypos ~b (~b ~b)", [Action, ItemID, PartyPos, A, B]),
%% @todo TargetGID and TargetLID must be validated, they're either the player's or his NPC characters.
event({item_equip, ItemID, TargetGID, TargetLID, A, B}) ->
GID = get(gid),
Category = case ItemID of
% units would be 8, traps would be 12
@ -572,41 +571,48 @@ handle(16#0105, Data) ->
Y when Y =:= 5; Y =:= 6; Y =:= 7 -> 0; % clothes
_ -> 1 % weapons
end,
case Action of
1 -> % equip item
Filename = case ItemID of
% weapons
16 -> "p/packet0105_sword.bin";
13 -> "p/packet0105_twindaggers.bin";
15 -> "p/packet0105_dagger.bin";
9 -> "p/packet0105_rcsm.bin";
14 -> "p/packet0105_saber.bin";
8 -> "p/packet0105_mgun.bin";
X when X =:= 17; X =:= 18 ->
"p/packet0105_twinguns.bin";
% armor
19 -> "p/packet0105_armor.bin";
% clothes
X when X =:= 5; X =:= 6; X =:= 7 ->
none;
_ -> % default: do nothing
none
end,
case Filename of
none -> File = << >>;
_ -> {ok, File} = file:read_file(Filename)
end,
send(<< 16#01050300:32, 0:64, GID:32/little-unsigned-integer, 0:64, 16#00011300:32, GID:32/little-unsigned-integer,
0:64, GID:32/little-unsigned-integer, PartyPos:32/little-unsigned-integer, ItemID, Action, Category, A, B:32/little-unsigned-integer,
File/binary >>);
2 -> % unequip item
send(<< 16#01050300:32, 0:64, GID:32/little-unsigned-integer, 0:64, 16#00011300:32, GID:32/little-unsigned-integer,
0:64, GID:32/little-unsigned-integer, PartyPos:32/little-unsigned-integer, ItemID, Action, Category, A, B:32/little-unsigned-integer >>);
5 -> % drop item
ignore;
_ ->
ignore
end;
Filename = case ItemID of
% weapons
16 -> "p/packet0105_sword.bin";
13 -> "p/packet0105_twindaggers.bin";
15 -> "p/packet0105_dagger.bin";
9 -> "p/packet0105_rcsm.bin";
14 -> "p/packet0105_saber.bin";
8 -> "p/packet0105_mgun.bin";
X when X =:= 17; X =:= 18 ->
"p/packet0105_twinguns.bin";
% armor
19 -> "p/packet0105_armor.bin";
% clothes
X when X =:= 5; X =:= 6; X =:= 7 ->
none;
_ -> % default, for lou
"p/packet0105_twindaggers.bin"
end,
case Filename of
none -> File = << >>;
_ -> {ok, File} = file:read_file(Filename)
end,
send(<< 16#01050300:32, 0:64, GID:32/little-unsigned-integer, 0:64, 16#00011300:32, GID:32/little-unsigned-integer,
0:64, TargetGID:32/little-unsigned-integer, TargetLID:32/little-unsigned-integer, ItemID, 1, Category, A, B:32/little-unsigned-integer,
File/binary >>);
%% @todo A and B are unknown.
%% @see item_equip
event({item_unequip, ItemID, TargetGID, TargetLID, A, B}) ->
GID = get(gid),
Category = case ItemID of
% units would be 8, traps would be 12
19 -> 2; % armor
Y when Y =:= 5; Y =:= 6; Y =:= 7 -> 0; % clothes
_ -> 1 % weapons
end,
send(<< 16#01050300:32, 0:64, GID:32/little-unsigned-integer, 0:64, 16#00011300:32, GID:32/little-unsigned-integer,
0:64, TargetGID:32/little-unsigned-integer, TargetLID:32/little-unsigned-integer, ItemID, 2, Category, A, B:32/little-unsigned-integer >>).
%% @doc Movement (non-broadcast) handler. Do nothing.
handle(16#0102, _) ->
ignore;
%% @doc Shop listing request. Currently return the normal item shop for everything.
%% @todo Return the other shops appropriately.

View File

@ -1,24 +1,106 @@
% EGS: Erlang Game Server
% Copyright (C) 2010 Loic Hoguin
%
% 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 <http://www.gnu.org/licenses/>.
%% @author Loïc Hoguin <essen@dev-extend.eu>
%% @copyright 2010 Loïc Hoguin.
%% @doc Independent implementation of the PSU protocol.
%%
%% 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 <http://www.gnu.org/licenses/>.
-module(psu_proto).
-compile(export_all).
%~ %% @todo We probably want to use active connections everywhere instead of doing this.
%~ recv %% remove later?
%~ %% @todo We probably want to remove this after all send functions are moved back in psu_proto.
%~ send %% fragments automatically if needed
%~ split
%% @doc Log the message.
log(Msg) ->
io:format("~p ~s~n", [get(gid), Msg]).
%% @spec log(Msg, FmtVars) -> ok
%% @doc Format and log the message.
log(Msg, FmtVars) ->
FmtMsg = io_lib:format(Msg, FmtVars),
log(FmtMsg).
%% @spec assert() -> ok
%% @doc Log a detailed message when the function is called.
-define(ASSERT(), log("assert error in module ~p on line ~p~n", [?MODULE, ?LINE])).
%% @spec assert(A, B) -> ok
%% @doc Log a detailed message when the assertion A =:= B fails.
-define(ASSERT_EQ(A, B), if A =:= B -> ok; true -> log("assert error in module ~p on line ~p~n", [?MODULE, ?LINE]) end).
%% @spec parse(Packet) -> Result
%% @doc Parse the packet and return a result accordingly.
parse(<< Size:32/little, Command:16, Channel:8, _Unknown:8, Data/bits >>) ->
parse(Size, Command, Channel, Data).
%% @todo One of the missing events is probably learning a new PA.
parse(Size, 16#0105, Channel, Data) ->
<< _VarA:16/little, _VarB:16/little, VarC:32/little, _FromGID:32/little, VarD:32/little, VarE:32/little, TypeID:32/little, GID:32/little,
VarF:32/little, VarG:32/little, TargetGID:32/little, TargetLID:32/little, ItemID:8, EventID:8, _PAID:8, VarH:8, VarI:32/little, Rest/bits >> = Data,
?ASSERT_EQ(Channel, 2),
?ASSERT_EQ(VarC, 0),
?ASSERT_EQ(VarD, 0),
?ASSERT_EQ(VarE, 0),
?ASSERT_EQ(TypeID, 0),
?ASSERT_EQ(GID, 0),
?ASSERT_EQ(VarF, 0),
?ASSERT_EQ(VarG, 0),
Event = case EventID of
1 -> item_equip;
2 -> item_unequip;
3 -> ignore; %% @todo item_link_pa;
4 -> ignore; %% @todo item_unlink_pa;
5 -> item_drop;
7 -> ?ASSERT(), ignore;
8 -> ignore; %% @todo item_use;
9 -> ?ASSERT(), ignore;
18 -> ignore; %% @todo item_unlearn_pa;
_ -> log("unknown 0105 EventID ~p", [EventID])
end,
case Event of
item_drop ->
?ASSERT_EQ(Size, 76),
<< _Quantity:32/little, _PosX:32/little-float, _PosY:32/little-float, _PosZ:32/little-float >> = Rest,
%~ {Event, ItemID, Quantity, ...};
ignore;
ignore ->
?ASSERT_EQ(Size, 60),
ignore;
_ ->
?ASSERT_EQ(Size, 60),
{Event, ItemID, TargetGID, TargetLID, VarH, VarI}
end;
parse(Size, 16#0b05, _Channel, _Data) ->
?ASSERT_EQ(Size, 8),
ignore;
parse(_Size, Command, Channel, Data) ->
%% @todo log unknown command?
%~ ignore.
<< _:288, Rest/bits >> = Data,
{command, Command, Channel, Rest}.
%% @doc Prepare a packet. Return the real size and padding at the end.
packet_prepare(Packet) ->