psu_proto: Clean parsing separation. Handle events rather than packets.
Lou can now equip a wepon.
This commit is contained in:
parent
f6305c3c76
commit
6dd159dc1e
@ -497,11 +497,16 @@ loop(SoFar) ->
|
|||||||
|
|
||||||
%% @doc Dispatch the command to the right handler.
|
%% @doc Dispatch the command to the right handler.
|
||||||
dispatch(Orig) ->
|
dispatch(Orig) ->
|
||||||
{command, Command, Channel, Data} = psu_proto:packet_parse(Orig),
|
case psu_proto:parse(Orig) of
|
||||||
|
{command, Command, Channel, Data} ->
|
||||||
case Channel of
|
case Channel of
|
||||||
ignore -> ignore;
|
|
||||||
1 -> broadcast(Command, Orig);
|
1 -> broadcast(Command, Orig);
|
||||||
_ -> handle(Command, Data)
|
_ -> handle(Command, Data)
|
||||||
|
end;
|
||||||
|
ignore ->
|
||||||
|
ignore;
|
||||||
|
Event ->
|
||||||
|
event(Event)
|
||||||
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.
|
||||||
@ -550,21 +555,15 @@ broadcast(Command, Orig)
|
|||||||
lists:foreach(fun(User) -> User#egs_user_model.pid ! {psu_broadcast, Packet} end, SpawnList)
|
lists:foreach(fun(User) -> User#egs_user_model.pid ! {psu_broadcast, Packet} end, SpawnList)
|
||||||
end.
|
end.
|
||||||
|
|
||||||
%% @doc Movement (non-broadcast) handler. Do nothing.
|
%% @todo A and B are unknown.
|
||||||
handle(16#0102, _) ->
|
|
||||||
ignore;
|
|
||||||
|
|
||||||
%% @doc Weapon equip, unequip, item drop, and more... handler. Do what we can.
|
|
||||||
%% Melee uses a format similar to: AAAA--BBCCCC----DDDDDDDDEE----FF with
|
%% 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.
|
%% 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.
|
%% 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 Others probably want to see that you changed your weapon.
|
||||||
%% @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.
|
||||||
%% @todo We must also handle here the NPC characters. PartyPos can be used for that, and more info in unknown values maybe too?
|
%% @todo TargetGID and TargetLID must be validated, they're either the player's or his NPC characters.
|
||||||
handle(16#0105, Data) ->
|
event({item_equip, ItemID, TargetGID, TargetLID, A, B}) ->
|
||||||
<< _: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]),
|
|
||||||
GID = get(gid),
|
GID = get(gid),
|
||||||
Category = case ItemID of
|
Category = case ItemID of
|
||||||
% units would be 8, traps would be 12
|
% units would be 8, traps would be 12
|
||||||
@ -572,8 +571,6 @@ handle(16#0105, Data) ->
|
|||||||
Y when Y =:= 5; Y =:= 6; Y =:= 7 -> 0; % clothes
|
Y when Y =:= 5; Y =:= 6; Y =:= 7 -> 0; % clothes
|
||||||
_ -> 1 % weapons
|
_ -> 1 % weapons
|
||||||
end,
|
end,
|
||||||
case Action of
|
|
||||||
1 -> % equip item
|
|
||||||
Filename = case ItemID of
|
Filename = case ItemID of
|
||||||
% weapons
|
% weapons
|
||||||
16 -> "p/packet0105_sword.bin";
|
16 -> "p/packet0105_sword.bin";
|
||||||
@ -589,24 +586,33 @@ handle(16#0105, Data) ->
|
|||||||
% clothes
|
% clothes
|
||||||
X when X =:= 5; X =:= 6; X =:= 7 ->
|
X when X =:= 5; X =:= 6; X =:= 7 ->
|
||||||
none;
|
none;
|
||||||
_ -> % default: do nothing
|
_ -> % default, for lou
|
||||||
none
|
"p/packet0105_twindaggers.bin"
|
||||||
end,
|
end,
|
||||||
case Filename of
|
case Filename of
|
||||||
none -> File = << >>;
|
none -> File = << >>;
|
||||||
_ -> {ok, File} = file:read_file(Filename)
|
_ -> {ok, File} = file:read_file(Filename)
|
||||||
end,
|
end,
|
||||||
send(<< 16#01050300:32, 0:64, GID:32/little-unsigned-integer, 0:64, 16#00011300:32, GID:32/little-unsigned-integer,
|
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,
|
0:64, TargetGID:32/little-unsigned-integer, TargetLID:32/little-unsigned-integer, ItemID, 1, Category, A, B:32/little-unsigned-integer,
|
||||||
File/binary >>);
|
File/binary >>);
|
||||||
2 -> % unequip item
|
|
||||||
|
%% @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,
|
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 >>);
|
0:64, TargetGID:32/little-unsigned-integer, TargetLID:32/little-unsigned-integer, ItemID, 2, Category, A, B:32/little-unsigned-integer >>).
|
||||||
5 -> % drop item
|
|
||||||
|
%% @doc Movement (non-broadcast) handler. Do nothing.
|
||||||
|
handle(16#0102, _) ->
|
||||||
ignore;
|
ignore;
|
||||||
_ ->
|
|
||||||
ignore
|
|
||||||
end;
|
|
||||||
|
|
||||||
%% @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.
|
||||||
|
@ -1,24 +1,106 @@
|
|||||||
% EGS: Erlang Game Server
|
%% @author Loïc Hoguin <essen@dev-extend.eu>
|
||||||
% Copyright (C) 2010 Loic Hoguin
|
%% @copyright 2010 Loïc Hoguin.
|
||||||
%
|
%% @doc Independent implementation of the PSU protocol.
|
||||||
% This file is part of EGS.
|
%%
|
||||||
%
|
%% 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
|
%% EGS is free software: you can redistribute it and/or modify
|
||||||
% the Free Software Foundation, either version 3 of the License, or
|
%% it under the terms of the GNU General Public License as published by
|
||||||
% (at your option) any later version.
|
%% 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
|
%% EGS is distributed in the hope that it will be useful,
|
||||||
% MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
%% but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
% GNU General Public License for more details.
|
%% 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/>.
|
%% 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).
|
-module(psu_proto).
|
||||||
-compile(export_all).
|
-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.
|
%% @doc Prepare a packet. Return the real size and padding at the end.
|
||||||
|
|
||||||
packet_prepare(Packet) ->
|
packet_prepare(Packet) ->
|
||||||
|
Loading…
Reference in New Issue
Block a user