/* * Copyright 2005-2007, Ingo Weinhold, bonefish@users.sf.net. All rights reserved. * Distributed under the terms of the MIT License. */ #ifndef KMESSAGE_H #define KMESSAGE_H #include #include #include #ifdef __cplusplus class BMessage; namespace BPrivate { class KMessageField; class MessageAdapter; // KMessage class KMessage { public: enum { KMESSAGE_OWNS_BUFFER = 0x01, KMESSAGE_INIT_FROM_BUFFER = 0x02, KMESSAGE_READ_ONLY = 0x04, KMESSAGE_FLAG_MASK = 0x07, }; KMessage(); KMessage(uint32 what); ~KMessage(); status_t SetTo(uint32 what, uint32 flags = 0); status_t SetTo(void *buffer, int32 bufferSize, uint32 what, uint32 flags = 0); status_t SetTo(const void *buffer, int32 bufferSize = -1); void Unset(); void SetWhat(uint32 what); uint32 What() const; const void *Buffer() const; int32 BufferCapacity() const; int32 ContentSize() const; status_t AddField(const char *name, type_code type, int32 elementSize = -1, KMessageField *field = NULL); status_t FindField(const char *name, KMessageField *field) const; status_t FindField(const char *name, type_code type, KMessageField *field) const; status_t GetNextField(KMessageField *field) const; status_t AddData(const char *name, type_code type, const void *data, int32 numBytes, bool isFixedSize = true); status_t AddArray(const char *name, type_code type, const void *data, int32 elementSize, int32 elementCount); inline status_t AddBool(const char *name, bool value); inline status_t AddInt8(const char *name, int8 value); inline status_t AddInt16(const char *name, int16 value); inline status_t AddInt32(const char *name, int32 value); inline status_t AddInt64(const char *name, int64 value); inline status_t AddString(const char *name, const char *value); status_t FindData(const char *name, type_code type, const void **data, int32 *numBytes) const; status_t FindData(const char *name, type_code type, int32 index, const void **data, int32 *numBytes) const; inline status_t FindBool(const char *name, bool *value) const; inline status_t FindBool(const char *name, int32 index, bool *value) const; inline status_t FindInt8(const char *name, int8 *value) const; inline status_t FindInt8(const char *name, int32 index, int8 *value) const; inline status_t FindInt16(const char *name, int16 *value) const; inline status_t FindInt16(const char *name, int32 index, int16 *value) const; inline status_t FindInt32(const char *name, int32 *value) const; inline status_t FindInt32(const char *name, int32 index, int32 *value) const; inline status_t FindInt64(const char *name, int64 *value) const; inline status_t FindInt64(const char *name, int32 index, int64 *value) const; inline status_t FindString(const char *name, const char **value) const; inline status_t FindString(const char *name, int32 index, const char **value) const; inline bool GetBool(const char* name, bool defaultValue) const; inline bool GetBool(const char* name, int32 index, bool defaultValue) const; inline int8 GetInt8(const char* name, int8 defaultValue) const; inline int8 GetInt8(const char* name, int32 index, int8 defaultValue) const; inline int16 GetInt16(const char* name, int16 defaultValue) const; inline int16 GetInt16(const char* name, int32 index, int16 defaultValue) const; inline int32 GetInt32(const char* name, int32 defaultValue) const; inline int32 GetInt32(const char* name, int32 index, int32 defaultValue) const; inline int64 GetInt64(const char* name, int64 defaultValue) const; inline int64 GetInt64(const char* name, int32 index, int64 defaultValue) const; inline const char* GetString(const char* name, const char* defaultValue) const; inline const char* GetString(const char* name, int32 index, const char* defaultValue) const; // fixed size fields only status_t SetData(const char* name, type_code type, const void* data, int32 numBytes); inline status_t SetBool(const char* name, bool value); inline status_t SetInt8(const char* name, int8 value); inline status_t SetInt16(const char* name, int16 value); inline status_t SetInt32(const char* name, int32 value); inline status_t SetInt64(const char* name, int64 value); // message delivery team_id Sender() const; int32 TargetToken() const; port_id ReplyPort() const; int32 ReplyToken() const; void SetDeliveryInfo(int32 targetToken, port_id replyPort, int32 replyToken, team_id senderTeam); status_t SendTo(port_id targetPort, int32 targetToken = -1, port_id replyPort = -1, int32 replyToken = -1, bigtime_t timeout = -1, team_id senderTeam = -1); status_t SendTo(port_id targetPort, int32 targetToken, KMessage* reply, bigtime_t deliveryTimeout = -1, bigtime_t replyTimeout = -1, team_id senderTeam = -1); status_t SendReply(KMessage* message, port_id replyPort = -1, int32 replyToken = -1, bigtime_t timeout = -1, team_id senderTeam = -1); status_t SendReply(KMessage* message, KMessage* reply, bigtime_t deliveryTimeout = -1, bigtime_t replyTimeout = -1, team_id senderTeam = -1); status_t ReceiveFrom(port_id fromPort, bigtime_t timeout = -1, port_message_info* messageInfo = NULL); void Dump(void (*printFunc)(const char*,...)); private: friend class KMessageField; friend class MessageAdapter; friend class ::BMessage; // not so nice, but makes things easier struct Header { uint32 magic; int32 size; uint32 what; team_id sender; int32 targetToken; port_id replyPort; int32 replyToken; }; struct FieldHeader; struct FieldValueHeader; Header *_Header() const; int32 _BufferOffsetFor(const void* data) const; FieldHeader *_FirstFieldHeader() const; FieldHeader *_LastFieldHeader() const; FieldHeader *_FieldHeaderForOffset(int32 offset) const; status_t _AddField(const char *name, type_code type, int32 elementSize, KMessageField *field); status_t _AddFieldData(KMessageField *field, const void *data, int32 elementSize, int32 elementCount); status_t _InitFromBuffer(bool sizeFromBuffer); void _InitBuffer(uint32 what); void _CheckBuffer(); // debugging only status_t _AllocateSpace(int32 size, bool alignAddress, bool alignSize, void **address, int32 *alignedSize); int32 _CapacityFor(int32 size); template inline status_t _FindType(const char* name, type_code type, int32 index, T *value) const; template inline T _GetType(const char* name, type_code type, int32 index, const T& defaultValue) const; Header fHeader; // pointed to by fBuffer, if nothing is // allocated void* fBuffer; int32 fBufferCapacity; uint32 fFlags; int32 fLastFieldOffset; static const uint32 kMessageHeaderMagic; }; // KMessageField class KMessageField { public: KMessageField(); void Unset(); KMessage *Message() const; const char *Name() const; type_code TypeCode() const; bool HasFixedElementSize() const; int32 ElementSize() const; // if HasFixedElementSize() status_t AddElement(const void *data, int32 size = -1); status_t AddElements(const void *data, int32 count, int32 elementSize = -1); const void *ElementAt(int32 index, int32 *size = NULL) const; int32 CountElements() const; private: void SetTo(KMessage *message, int32 headerOffset); KMessage::FieldHeader* _Header() const; friend class KMessage; KMessage *fMessage; int32 fHeaderOffset; }; } // namespace BPrivate using BPrivate::KMessage; using BPrivate::KMessageField; // #pragma mark - inline functions // AddBool inline status_t KMessage::AddBool(const char *name, bool value) { return AddData(name, B_BOOL_TYPE, &value, sizeof(bool), true); } // AddInt8 inline status_t KMessage::AddInt8(const char *name, int8 value) { return AddData(name, B_INT8_TYPE, &value, sizeof(int8), true); } // AddInt16 inline status_t KMessage::AddInt16(const char *name, int16 value) { return AddData(name, B_INT16_TYPE, &value, sizeof(int16), true); } // AddInt32 inline status_t KMessage::AddInt32(const char *name, int32 value) { return AddData(name, B_INT32_TYPE, &value, sizeof(int32), true); } // AddInt64 inline status_t KMessage::AddInt64(const char *name, int64 value) { return AddData(name, B_INT64_TYPE, &value, sizeof(int64), true); } // AddString inline status_t KMessage::AddString(const char *name, const char *value) { if (!value) return B_BAD_VALUE; return AddData(name, B_STRING_TYPE, value, strlen(value) + 1, false); } // #pragma mark - // _FindType template inline status_t KMessage::_FindType(const char* name, type_code type, int32 index, T *value) const { const void *data; int32 size; status_t error = FindData(name, type, index, &data, &size); if (error != B_OK) return error; if (size != sizeof(T)) return B_BAD_DATA; *value = *(T*)data; return B_OK; } // FindBool inline status_t KMessage::FindBool(const char *name, bool *value) const { return FindBool(name, 0, value); } // FindBool inline status_t KMessage::FindBool(const char *name, int32 index, bool *value) const { return _FindType(name, B_BOOL_TYPE, index, value); } // FindInt8 inline status_t KMessage::FindInt8(const char *name, int8 *value) const { return FindInt8(name, 0, value); } // FindInt8 inline status_t KMessage::FindInt8(const char *name, int32 index, int8 *value) const { return _FindType(name, B_INT8_TYPE, index, value); } // FindInt16 inline status_t KMessage::FindInt16(const char *name, int16 *value) const { return FindInt16(name, 0, value); } // FindInt16 inline status_t KMessage::FindInt16(const char *name, int32 index, int16 *value) const { return _FindType(name, B_INT16_TYPE, index, value); } // FindInt32 inline status_t KMessage::FindInt32(const char *name, int32 *value) const { return FindInt32(name, 0, value); } // FindInt32 inline status_t KMessage::FindInt32(const char *name, int32 index, int32 *value) const { return _FindType(name, B_INT32_TYPE, index, value); } // FindInt64 inline status_t KMessage::FindInt64(const char *name, int64 *value) const { return FindInt64(name, 0, value); } // FindInt64 inline status_t KMessage::FindInt64(const char *name, int32 index, int64 *value) const { return _FindType(name, B_INT64_TYPE, index, value); } // FindString inline status_t KMessage::FindString(const char *name, const char **value) const { return FindString(name, 0, value); } // FindString inline status_t KMessage::FindString(const char *name, int32 index, const char **value) const { int32 size; return FindData(name, B_STRING_TYPE, index, (const void**)value, &size); } // _GetType template inline T KMessage::_GetType(const char* name, type_code type, int32 index, const T& defaultValue) const { T value; if (_FindType(name, type, index, &value) == B_OK) return value; return defaultValue; } // GetBool inline bool KMessage::GetBool(const char* name, bool defaultValue) const { return _GetType(name, B_BOOL_TYPE, 0, defaultValue); } // GetBool inline bool KMessage::GetBool(const char* name, int32 index, bool defaultValue) const { return _GetType(name, B_BOOL_TYPE, index, defaultValue); } // GetInt8 inline int8 KMessage::GetInt8(const char* name, int8 defaultValue) const { return _GetType(name, B_INT8_TYPE, 0, defaultValue); } // GetInt8 inline int8 KMessage::GetInt8(const char* name, int32 index, int8 defaultValue) const { return _GetType(name, B_INT8_TYPE, index, defaultValue); } // GetInt16 inline int16 KMessage::GetInt16(const char* name, int16 defaultValue) const { return _GetType(name, B_INT16_TYPE, 0, defaultValue); } // GetInt16 inline int16 KMessage::GetInt16(const char* name, int32 index, int16 defaultValue) const { return _GetType(name, B_INT16_TYPE, index, defaultValue); } // GetInt32 inline int32 KMessage::GetInt32(const char* name, int32 defaultValue) const { return _GetType(name, B_INT32_TYPE, 0, defaultValue); } // GetInt32 inline int32 KMessage::GetInt32(const char* name, int32 index, int32 defaultValue) const { return _GetType(name, B_INT32_TYPE, index, defaultValue); } // GetInt64 inline int64 KMessage::GetInt64(const char* name, int64 defaultValue) const { return _GetType(name, B_INT64_TYPE, 0, defaultValue); } // GetInt64 inline int64 KMessage::GetInt64(const char* name, int32 index, int64 defaultValue) const { return _GetType(name, B_INT64_TYPE, index, defaultValue); } // GetString inline const char* KMessage::GetString(const char* name, int32 index, const char* defaultValue) const { // don't use _GetType() here, since it checks field size == sizeof(T) int32 size; const char* value; if (FindData(name, B_STRING_TYPE, index, (const void**)&value, &size) == B_OK) { return value; } return defaultValue; } // GetString inline const char* KMessage::GetString(const char* name, const char* defaultValue) const { return GetString(name, 0, defaultValue); } // SetBool inline status_t KMessage::SetBool(const char* name, bool value) { return SetData(name, B_BOOL_TYPE, &value, sizeof(bool)); } // SetInt8 inline status_t KMessage::SetInt8(const char* name, int8 value) { return SetData(name, B_INT8_TYPE, &value, sizeof(int8)); } // SetInt16 inline status_t KMessage::SetInt16(const char* name, int16 value) { return SetData(name, B_INT16_TYPE, &value, sizeof(int16)); } // SetInt32 inline status_t KMessage::SetInt32(const char* name, int32 value) { return SetData(name, B_INT32_TYPE, &value, sizeof(int32)); } // SetInt64 inline status_t KMessage::SetInt64(const char* name, int64 value) { return SetData(name, B_INT64_TYPE, &value, sizeof(int64)); } #else // !__cplusplus typedef struct KMessage { struct Header { uint32 magic; int32 size; uint32 what; team_id sender; int32 targetToken; port_id replyPort; int32 replyToken; } fHeader; void* fBuffer; int32 fBufferCapacity; uint32 fFlags; int32 fLastFieldOffset; } KMessage; #endif // !__cplusplus #endif // KMESSAGE_H