DarkflameServer/thirdparty/raknet/Source/RakString.h
2021-12-05 18:54:36 +01:00

203 lines
6.2 KiB
C++

#ifndef __RAK_STRING_H
#define __RAK_STRING_H
#include "Export.h"
#include "DS_List.h"
#include "RakNetTypes.h" // int64_t
#include <stdio.h>
namespace RakNet
{
class BitStream;
/// \brief String class
/// Has the following improvements over std::string
/// Reference counting: Suitable to store in lists
/// Varidic assignment operator
/// Doesn't cause linker errors
class RAK_DLL_EXPORT RakString
{
public:
/// Constructors
RakString();
RakString(char input);
RakString(unsigned char input);
RakString(const unsigned char *format, ...);
RakString(const char *format, ...);
~RakString();
RakString( const RakString & rhs);
/// Implicit return of const char*
operator const char* () const {return sharedString->c_str;}
/// Same as std::string::c_str
const char *C_String(void) const {return sharedString->c_str;}
/// Assigment operators
RakString& operator = ( const RakString& rhs );
RakString& operator = ( const char *str );
RakString& operator = ( char *str );
RakString& operator = ( const char c );
/// Concatenation
RakString& operator +=( const RakString& rhs);
RakString& operator += ( const char *str );
RakString& operator += ( char *str );
RakString& operator += ( const char c );
/// Character index. Do not use to change the string however.
unsigned char operator[] ( const unsigned int position ) const;
/// Equality
bool operator==(const RakString &rhs) const;
bool operator==(const char *str) const;
bool operator==(char *str) const;
/// Inequality
bool operator!=(const RakString &rhs) const;
/// Change all characters to lowercase
void ToLower(void);
/// Change all characters to uppercase
void ToUpper(void);
/// Set the value of the string
void Set(const char *format, ...);
/// Returns if the string is empty. Also, C_String() would return ""
bool IsEmpty(void) const;
/// Returns the length of the string
size_t GetLength(void) const;
/// Replace character(s) in starting at index, for count, with c
void Replace(unsigned index, unsigned count, unsigned char c);
/// Erase characters out of the string at index for count
void Erase(unsigned index, unsigned count);
/// Compare strings (case sensitive)
int StrCmp(const RakString &rhs) const;
/// Compare strings (not case sensitive)
int StrICmp(const RakString &rhs) const;
/// Clear the string
void Clear(void);
/// Print the string to the screen
void Printf(void);
/// Print the string to a file
void FPrintf(FILE *fp);
/// Does the given IP address match the IP address encoded into this string, accounting for wildcards?
bool IPAddressMatch(const char *IP);
/// URL Encode the string. See http://www.codeguru.com/cpp/cpp/cpp_mfc/article.php/c4029/
void URLEncode(void);
/// RakString uses a freeList of old no-longer used strings
/// Call this function to clear this memory on shutdown
static void FreeMemory(void);
/// Serialize to a bitstream, uncompressed (slightly faster)
/// \param[out] bs Bitstream to serialize to
void Serialize(BitStream *bs);
/// Static version of the Serialize function
static void Serialize(const char *str, BitStream *bs);
/// Serialize to a bitstream, compressed (better bandwidth usage)
/// \param[out] bs Bitstream to serialize to
/// \param[in] languageId languageId to pass to the StringCompressor class
/// \param[in] writeLanguageId encode the languageId variable in the stream. If false, 0 is assumed, and DeserializeCompressed will not look for this variable in the stream (saves bandwidth)
/// \pre StringCompressor::AddReference must have been called to instantiate the class (Happens automatically from RakPeer::Startup())
void SerializeCompressed(BitStream *bs, int languageId=0, bool writeLanguageId=false);
/// Static version of the SerializeCompressed function
static void SerializeCompressed(const char *str, BitStream *bs, int languageId=0, bool writeLanguageId=false);
/// Deserialize what was written by Serialize
/// \param[in] bs Bitstream to serialize from
/// \return true if the deserialization was successful
bool Deserialize(BitStream *bs);
/// Static version of the Deserialize() function
static bool Deserialize(char *str, BitStream *bs);
/// Deserialize compressed string, written by SerializeCompressed
/// \param[in] bs Bitstream to serialize from
/// \param[in] readLanguageId If true, looks for the variable langaugeId in the data stream. Must match what was passed to SerializeCompressed
/// \return true if the deserialization was successful
/// \pre StringCompressor::AddReference must have been called to instantiate the class (Happens automatically from RakPeer::Startup())
bool DeserializeCompressed(BitStream *bs, bool readLanguageId=false);
/// Static version of the DeserializeCompressed() function
static bool DeserializeCompressed(char *str, BitStream *bs, bool readLanguageId=false);
static const char *ToString(int64_t i);
static const char *ToString(uint64_t i);
/// \internal
static size_t GetSizeToAllocate(size_t bytes)
{
const size_t smallStringSize = 128-sizeof(unsigned int)-sizeof(size_t)-sizeof(char*)*2;
if (bytes<=smallStringSize)
return smallStringSize;
else
return bytes*2;
}
/// \internal
struct SharedString
{
unsigned int refCount;
size_t bytesUsed;
char *bigString;
char *c_str;
char smallString[128-sizeof(unsigned int)-sizeof(size_t)-sizeof(char*)*2];
};
/// \internal
RakString( SharedString *_sharedString );
/// \internal
SharedString *sharedString;
// static SimpleMutex poolMutex;
// static DataStructures::MemoryPool<SharedString> pool;
/// \internal
static SharedString emptyString;
//static SharedString *sharedStringFreeList;
//static unsigned int sharedStringFreeListAllocationCount;
/// \internal
/// List of free objects to reduce memory reallocations
static DataStructures::List<SharedString*> freeList;
/// Means undefined position
static unsigned int nPos;
static int RakStringComp( RakString const &key, RakString const &data );
protected:
void Allocate(size_t len);
void Assign(const char *str);
void Clone(void);
void Free(void);
unsigned char ToLower(unsigned char c);
unsigned char ToUpper(unsigned char c);
void Realloc(SharedString *sharedString, size_t bytes);
};
}
const RakNet::RakString operator+(const RakNet::RakString &lhs, const RakNet::RakString &rhs);
#endif