/* * The ManaPlus Client * Copyright (C) 2004-2009 The Mana World Development Team * Copyright (C) 2009-2010 The Mana Developers * Copyright (C) 2011-2012 The ManaPlus Developers * * This file is part of The ManaPlus Client. * * This program 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 2 of the License, or * any later version. * * This program 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 this program. If not, see <http://www.gnu.org/licenses/>. */ #include "net/messagein.h" #include "net/packetcounters.h" #include "logger.h" #include "net.h" #include "utils/stringutils.h" #include "debug.h" #define MAKEWORD(low, high) \ (static_cast<unsigned short>((static_cast<unsigned char>(low)) | \ (static_cast<unsigned short>(static_cast<unsigned char>(high))) << 8)) namespace Net { MessageIn::MessageIn(const char *data, unsigned int length): mData(data), mLength(length), mId(0), mPos(0) { PacketCounters::incInPackets(); DEBUGLOG("MessageIn"); } unsigned char MessageIn::readInt8() { unsigned char value = -1; if (mPos < mLength) value = static_cast<unsigned char>(mData[mPos]); mPos += 1; PacketCounters::incInBytes(1); DEBUGLOG("readInt8: " + toString(static_cast<int>(value))); return value; } void MessageIn::readCoordinates(Uint16 &x, Uint16 &y) { if (mPos + 3 <= mLength) { unsigned char const *p = reinterpret_cast< unsigned char const * >(mData + mPos); x = static_cast<short unsigned>(p[0] | ((p[1] & 0x07) << 8)); y = static_cast<short unsigned>((p[1] >> 3) | ((p[2] & 0x3F) << 5)); } mPos += 3; PacketCounters::incInBytes(3); DEBUGLOG("readCoordinates: " + toString(static_cast<int>(x)) + "," + toString(static_cast<int>(y))); } void MessageIn::readCoordinates(Uint16 &x, Uint16 &y, Uint8 &direction) { Uint8 serverDir = 0; if (mPos + 3 <= mLength) { const char *data = mData + mPos; Sint16 temp; temp = MAKEWORD(data[1] & 0x00c0, data[0] & 0x00ff); x = static_cast<unsigned short>(temp >> 6); temp = MAKEWORD(data[2] & 0x00f0, data[1] & 0x003f); y = static_cast<unsigned short>(temp >> 4); serverDir = data[2] & 0x000f; // Translate from eAthena format switch (serverDir) { case 0: direction = 1; break; case 1: direction = 3; break; case 2: direction = 2; break; case 3: direction = 6; break; case 4: direction = 4; break; case 5: direction = 12; break; case 6: direction = 8; break; case 7: direction = 9; break; case 8: #ifdef MANASERV_SUPPORT if (Net::getNetworkType() != ServerInfo::MANASERV) #endif { direction = 8; break; } default: logger->log("incorrect direction: %d", static_cast<int>(direction)); // OOPSIE! Impossible or unknown direction = 0; } } mPos += 3; PacketCounters::incInBytes(3); DEBUGLOG("readCoordinates: " + toString(static_cast<int>(x)) + "," + toString(static_cast<int>(y)) + "," + toString( static_cast<int>(serverDir))); } void MessageIn::readCoordinatePair(Uint16 &srcX, Uint16 &srcY, Uint16 &dstX, Uint16 &dstY) { if (mPos + 5 <= mLength) { const char *data = mData + mPos; Sint16 temp; temp = MAKEWORD(data[3], data[2] & 0x000f); dstX = static_cast<unsigned short>(temp >> 2); dstY = MAKEWORD(data[4], data[3] & 0x0003); temp = MAKEWORD(data[1], data[0]); srcX = static_cast<unsigned short>(temp >> 6); temp = MAKEWORD(data[2], data[1] & 0x003f); srcY = static_cast<unsigned short>(temp >> 4); } mPos += 5; DEBUGLOG("readCoordinatePair: " + toString(static_cast<int>(srcX)) + "," + toString(static_cast<int>(srcY)) + " " + toString(static_cast<int>(dstX)) + "," + toString(static_cast<int>(dstY))); PacketCounters::incInBytes(5); } void MessageIn::skip(unsigned int length) { mPos += length; PacketCounters::incInBytes(length); DEBUGLOG("skip: " + toString(static_cast<int>(length))); } std::string MessageIn::readString(int length) { // Get string length if (length < 0) length = readInt16(); // Make sure the string isn't erroneous if (length < 0 || mPos + length > mLength) { mPos = mLength + 1; DEBUGLOG("readString error"); return ""; } // Read the string char const *stringBeg = mData + mPos; char const *stringEnd = static_cast<char const *>(memchr(stringBeg, '\0', length)); std::string str(stringBeg, stringEnd ? stringEnd - stringBeg : length); mPos += length; PacketCounters::incInBytes(length); DEBUGLOG("readString: " + str); return str; } std::string MessageIn::readRawString(int length) { // Get string length if (length < 0) length = readInt16(); // Make sure the string isn't erroneous if (length < 0 || mPos + length > mLength) { mPos = mLength + 1; return ""; } // Read the string char const *stringBeg = mData + mPos; char const *stringEnd = static_cast<char const *>(memchr(stringBeg, '\0', length)); std::string str(stringBeg, stringEnd ? stringEnd - stringBeg : length); mPos += length; PacketCounters::incInBytes(length); DEBUGLOG("readString: " + str); if (stringEnd) { long len2 = length - (stringEnd - stringBeg) - 1; char const *stringBeg2 = stringEnd + 1; char const *stringEnd2 = static_cast<char const *>(memchr(stringBeg2, '\0', len2)); std::string hiddenPart = std::string(stringBeg2, stringEnd2 ? stringEnd2 - stringBeg2 : len2); if (hiddenPart.length() > 0) { DEBUGLOG("readString2: " + hiddenPart); return str + "|" + hiddenPart; } } return str; } unsigned char *MessageIn::readBytes(int length) { // Get string length if (length < 0) length = readInt16(); // Make sure the string isn't erroneous if (length < 0 || mPos + length > mLength) { mPos = mLength + 1; DEBUGLOG("readBytesString error"); return nullptr; } unsigned char *buf = new unsigned char[length + 2]; memcpy (buf, mData + mPos, length); buf[length] = 0; buf[length + 1] = 0; mPos += length; #ifdef ENABLEDEBUGLOG std::string str; for (int f = 0;f < length; f ++) str += strprintf ("%02x", (unsigned)buf[f]); str += " "; for (int f = 0;f < length; f ++) { if (buf[f]) str += strprintf ("%c", buf[f]); else str += "_"; } logger->log("ReadBytes: " + str); #endif PacketCounters::incInBytes(length); return buf; } }