-
Notifications
You must be signed in to change notification settings - Fork 1.1k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Parse packet in lua feature #2978
Conversation
There is a "major" mistake in this pr in the protocolgame code. The player methods and specially the creatureevents code should be called by the dispatcher thread, you are calling them from the main/asio thread = crash. |
I guess this is not very hard to correct. However, I prefer separating this feature into modules, like in OTX. |
@cristofermartins interesting, can you tell me in which cases or why would it crash? |
@infister to get module "feeling" you can use revscriptsys. |
try to reload this with syntax error and then receive such packet
I am not talking about feeling, this just, in my opinion, should be separated from creaturescripts. |
You have a racing condition inside "getParsePacketEvent" if somehow new event will be registered from dispatcher thread and asio thread will be looping through eventsList it'll result in crash. Also I'm seeing that many people don't understand what is passing object by reference because your function "executeParsePacket" might get different data when it actually get called by dispatcher thread(data can be changed by asio thread). |
I prefer to use the OTX method, and it is also compatible with Revscripts ;) |
@MillhioreBT "otx" you meant @slavidodo method right? cause their code is stolen from slavi |
@SaiyansKing can you enligthen me, how and why would be the data changed by asio thread? I will agree thats something I dont know much so I want to learn, will enjoy a good explanation |
It's pretty good anyway, why switch to yours? is better? I would gladly change if necessary. |
@MillhioreBT ofc, you are free to use whatever you like to use. I'm not the one to judge which one is better and if you like their implementation better then use the one that suit you the best. |
Its just that when you had reference to "Connection::msg" asio thread would read new data to that buffer without waiting for your dispatcher task to do its work. It is rather rare racing condition but a possibility is a possibility so it was worth mentioning it. |
@SaiyansKing thanks! Can I request your review on that pr? I changed the code a bit I guess it should be fine now? |
I also have a question(which is related to this pull request): why the packet parsing is not done by the dispatcher thread? |
@cristofermartins can you review the pull request, because I'm not sure what you mean now? |
If you're asking me then I don't know performance-wise it doesn't matter because at most you read only few bytes and main job still do dispatcher, maybe they didn't want to make unnecessary copies of networkmessage buffer because as I posted before passing it as reference would have rare racing condition. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
It looks ok to me.
Memory leak in the Game::playerExecuteParsePacketEvent, message(pointer to NetworkMessage) is not deleted. |
@cristofermartins message is deleted in lua, its the same thing like when you use following code in lua and network message class in lua got garbage collector which calls delete
|
@nekiro Will a situation be possible where it is necessary to keep the pointer alive for an extra time at the end of the event? otherwise, we can clean pointer from fonts to avoid writing an additional line, what do you think? |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
It convinced me, I tested it and it seems to work very well.
@MillhioreBT you shouldnt have to delete the message manually, garbage collector should delete it at the next run, when reference count hits 0 so last msg:delete() isnt necessary, but better safe than sorry. |
In my opinion we should implement code from otservbr as its much better. |
@kygov if you want to openly say something is better than something else. You should give us arguments on why. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I agree with this, and might even insist on creating multiple instances of this event:
I did it this way and it worked wonderfully:
game.cpp
void Game::playerExecuteParsePacketEvent(uint32_t playerId, uint8_t recvbyte, NetworkMessage* message)
{
Player* player = getPlayerByID(playerId);
if (!player) {
return;
}
if (!player->hasEventRegistered(CREATURE_EVENT_PARSE_PACKET)) {
return;
}
for (CreatureEvent* creatureEvent : player->eventsList) {
if (!creatureEvent->isLoaded()) {
continue;
}
if (creatureEvent->getEventType() == CREATURE_EVENT_PARSE_PACKET && creatureEvent->getRecvbyte() == recvbyte) {
std::unique_ptr<NetworkMessage> msgPtr(new NetworkMessage(*message));
creatureEvent->executeParsePacket(player, recvbyte, std::move(msgPtr));
}
}
delete message;
message = nullptr;
}
Use this Revscript to verify that what I say is true:
data/scripts/parsepacket.lua
local parsePacketEvent = CreatureEvent("ParsePacket")
parsePacketEvent.onParsePacket = function (player, recvbyte, msg)
-- structure for parseUseItem packet, recbyte: 130 (0x82)
local pos = msg:getPosition()
local clientId = msg:getU16()
local stackpos = msg:getByte()
local index = msg:getByte()
print(("Player: %s wants to use item at position (%d, %d, %d) with sprite id %d"):format(player:getName(), pos.x, pos.y, pos.z, clientId))
msg:delete()
end
parsePacketEvent:recvbyte(130)
parsePacketEvent:register()
local cliportEvent = CreatureEvent("CliportLua")
cliportEvent.onParsePacket = function (player, recvbyte, msg)
local cliportStatus = player:getStorageValue(131070) == 1
if not cliportStatus then
return
end
local pos = msg:getPosition()
local path = player:getPathTo(pos, 0, 1, false, false, 0)
if path then
local fromPos = player:getPosition()
for i, dir in pairs(path) do
fromPos:getNextPosition(dir)
end
player:teleportTo(fromPos, true)
end
msg:delete()
end
cliportEvent:recvbyte(130)
cliportEvent:register()
local loginEvent = CreatureEvent("LoginEventCliport")
loginEvent.onLogin = function (player)
player:registerEvent("ParsePacket")
player:registerEvent("CliportLua")
return true
end
loginEvent:register()
local talkEvent = TalkAction("/cliport")
talkEvent.onSay = function (player, words, param)
if not player:getGroup():getAccess() then
return true
end
local cliportStatus = player:getStorageValue(131070) == -1
player:setStorageValue(131070, cliportStatus and 1 or -1)
if cliportStatus then
player:sendTextMessage(MESSAGE_EVENT_DEFAULT, "Cliport condition enabled.")
else
player:sendTextMessage(MESSAGE_EVENT_DEFAULT, "Cliport condition disabled.")
end
return false
end
talkEvent:register()
It seems to work too well without my small changes and also with them.
I was going through the code again and I just got an idea, what about events interface. This pr is currently using creaturescripts, but events looks like a better place for that? What do you think? |
If you could add multiple events for parsing different recvbytes, then it might be okay. |
I like the idea, and now that the PR #2867 is working perfectly, this has a very nice potential. |
Co-authored-by: Damian Jarek <[email protected]>
Hello, I want to revive this PR and suggest that it move to |
Smart pointers are made to be used. |
@EvilHero90 remove your stale pending review please |
Wouldn't it be better in events? |
Well probably yes, but also no. It's nice to have control over what you register to players on the other hand events is not a bad idea, though we would have to implement a class to parse these opcodes and store them to know which opcodes are registered in lua, or just store them in container in game class. |
You can just if + return on players you don't want to serve. |
@@ -44,6 +44,10 @@ class NetworkMessage | |||
enum { MAX_PROTOCOL_BODY_LENGTH = MAX_BODY_LENGTH - 10 }; | |||
|
|||
NetworkMessage() = default; | |||
NetworkMessage(const NetworkMessage& other) { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This doesn't make sense if you don't really plan to make a copy, I see that you modified PR to not make copies, so why this?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
future proof
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@@ -526,6 +526,11 @@ void ProtocolGame::parsePacket(NetworkMessage& msg) | |||
} | |||
} | |||
|
|||
if (player->hasEventRegistered(CREATURE_EVENT_PARSE_PACKET)) { | |||
addGameTask(&Game::playerExecuteParsePacketEvent, player->getID(), recvbyte, msg); | |||
return; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
this is serious? or a joke? XD
the return;
<<<
player:registerEvent("parsepacket")
>>> the player will stop doing things, the (all)-packet's are simply never parsed in the sources
This pr allows you to parse packet in lua using creaturescripts interface.
I have included template creaturescript how to parse packet.
Registering a packet in lua means it will not be parsed in source code.