diff options
| author | Mike Buland <eichlan@xagasoft.com> | 2011-01-10 21:04:17 +0000 | 
|---|---|---|
| committer | Mike Buland <eichlan@xagasoft.com> | 2011-01-10 21:04:17 +0000 | 
| commit | 2ba3f84ab559da02a11aa000b3cecb3b3668af61 (patch) | |
| tree | 266f450b512f607ec54d54af4fa8c13fdbe7ef91 /src | |
| parent | ea18007633b31901f2ae275cc0576c3f7ce99fc9 (diff) | |
| parent | 3611f253f6fdfa4954d374ab85ddaa7f799c130c (diff) | |
| download | libbu++-2ba3f84ab559da02a11aa000b3cecb3b3668af61.tar.gz libbu++-2ba3f84ab559da02a11aa000b3cecb3b3668af61.tar.bz2 libbu++-2ba3f84ab559da02a11aa000b3cecb3b3668af61.tar.xz libbu++-2ba3f84ab559da02a11aa000b3cecb3b3668af61.zip | |
Merged in the core branch.  This is a major update that fixes many things, and
changes many others, including source files that were deleted and renamed.
Before doing this update, I reccomend a full clean, or even a fresh checkout.
Things to note, most outstanding about this update:
 - Bu::Socket was changed to Bu::TcpSocket and the default mode is blocking.
 - All templatized container classes are SharedCore now, which is good, but
   SharedCore is inherently non-reentrant safe.  However, all SharedCore classes
   have a "clone" function that return a non-shared copy of the object, safe for
   passing into a reentrant safe function accessing shared memory.
Diffstat (limited to '')
| -rw-r--r-- | src/archive.cpp | 9 | ||||
| -rw-r--r-- | src/archive.h | 4 | ||||
| -rw-r--r-- | src/archivebase.h | 5 | ||||
| -rw-r--r-- | src/array.h | 29 | ||||
| -rw-r--r-- | src/cachestorefiles.h | 2 | ||||
| -rw-r--r-- | src/cachestoremyriad.h | 2 | ||||
| -rw-r--r-- | src/client.cpp | 9 | ||||
| -rw-r--r-- | src/client.h | 8 | ||||
| -rw-r--r-- | src/fastcgi.cpp | 22 | ||||
| -rw-r--r-- | src/fastcgi.h | 18 | ||||
| -rw-r--r-- | src/fbasicstring.h | 241 | ||||
| -rw-r--r-- | src/file.cpp | 1 | ||||
| -rw-r--r-- | src/hash.h | 1057 | ||||
| -rw-r--r-- | src/heap.h | 300 | ||||
| -rw-r--r-- | src/httpget.h | 4 | ||||
| -rw-r--r-- | src/itoserver.cpp | 12 | ||||
| -rw-r--r-- | src/itoserver.h | 6 | ||||
| -rw-r--r-- | src/lexer.h | 4 | ||||
| -rw-r--r-- | src/list.h | 29 | ||||
| -rw-r--r-- | src/listhash.cpp | 8 | ||||
| -rw-r--r-- | src/listhash.h | 54 | ||||
| -rw-r--r-- | src/myriad.h | 2 | ||||
| -rw-r--r-- | src/myriadfs.cpp | 78 | ||||
| -rw-r--r-- | src/myriadfs.h | 54 | ||||
| -rw-r--r-- | src/newline.h | 2 | ||||
| -rw-r--r-- | src/paramproc.cpp | 523 | ||||
| -rw-r--r-- | src/paramproc.h | 163 | ||||
| -rw-r--r-- | src/parser.cpp | 34 | ||||
| -rw-r--r-- | src/parser.h | 5 | ||||
| -rw-r--r-- | src/ringbuffer.h | 209 | ||||
| -rw-r--r-- | src/server.cpp | 14 | ||||
| -rw-r--r-- | src/server.h | 6 | ||||
| -rw-r--r-- | src/set.h | 760 | ||||
| -rw-r--r-- | src/sharedcore.h | 77 | ||||
| -rw-r--r-- | src/tcpserversocket.cpp (renamed from src/serversocket.cpp) | 36 | ||||
| -rw-r--r-- | src/tcpserversocket.h (renamed from src/serversocket.h) | 18 | ||||
| -rw-r--r-- | src/tcpsocket.cpp (renamed from src/socket.cpp) | 148 | ||||
| -rw-r--r-- | src/tcpsocket.h (renamed from src/socket.h) | 31 | ||||
| -rw-r--r-- | src/tests/sharedcore.cpp | 2 | ||||
| -rw-r--r-- | src/tests/socketblock.cpp | 10 | ||||
| -rw-r--r-- | src/tests/socketbreak.cpp | 10 | ||||
| -rw-r--r-- | src/tests/tcpsocket.cpp (renamed from src/tests/socket.cpp) | 4 | ||||
| -rw-r--r-- | src/tools/bnfcompile.cpp | 410 | ||||
| -rw-r--r-- | src/tools/myriadfs.cpp | 38 | ||||
| -rw-r--r-- | src/tools/parser.cpp | 89 | ||||
| -rw-r--r-- | src/unit/fstring.unit | 9 | ||||
| -rw-r--r-- | src/unit/hash.unit | 13 | ||||
| -rw-r--r-- | src/utfstring.cpp | 29 | ||||
| -rw-r--r-- | src/utfstring.h | 24 | ||||
| -rw-r--r-- | src/variant.h | 18 | 
50 files changed, 2163 insertions, 2477 deletions
| diff --git a/src/archive.cpp b/src/archive.cpp index 69d4a0c..7a10921 100644 --- a/src/archive.cpp +++ b/src/archive.cpp | |||
| @@ -22,20 +22,15 @@ Bu::Archive::~Archive() | |||
| 22 | { | 22 | { | 
| 23 | } | 23 | } | 
| 24 | 24 | ||
| 25 | void Bu::Archive::write( const void *pData, int32_t nSize ) | 25 | void Bu::Archive::write( const void *pData, size_t nSize ) | 
| 26 | { | 26 | { | 
| 27 | if( nSize == 0 || pData == NULL ) | 27 | if( nSize == 0 || pData == NULL ) | 
| 28 | return; | 28 | return; | 
| 29 | 29 | ||
| 30 | // Bu::sio << "Writing starting at pos: " << rStream.tell() << " - " | ||
| 31 | // << Bu::sio.flush; | ||
| 32 | |||
| 33 | rStream.write( (const char *)pData, nSize ); | 30 | rStream.write( (const char *)pData, nSize ); | 
| 34 | // | ||
| 35 | // Bu::sio << rStream.tell() << " (" << nSize << "b)" << Bu::sio.nl; | ||
| 36 | } | 31 | } | 
| 37 | 32 | ||
| 38 | void Bu::Archive::read( void *pData, int32_t nSize ) | 33 | void Bu::Archive::read( void *pData, size_t nSize ) | 
| 39 | { | 34 | { | 
| 40 | if( nSize == 0 || pData == NULL ) | 35 | if( nSize == 0 || pData == NULL ) | 
| 41 | return; | 36 | return; | 
| diff --git a/src/archive.h b/src/archive.h index bd85113..9d2aee2 100644 --- a/src/archive.h +++ b/src/archive.h | |||
| @@ -83,8 +83,8 @@ namespace Bu | |||
| 83 | virtual ~Archive(); | 83 | virtual ~Archive(); | 
| 84 | virtual void close(); | 84 | virtual void close(); | 
| 85 | 85 | ||
| 86 | virtual void write(const void *, int32_t); | 86 | virtual void write( const void *pData, size_t iSize ); | 
| 87 | virtual void read(void *, int32_t); | 87 | virtual void read( void *pData, size_t iSize ); | 
| 88 | 88 | ||
| 89 | /** | 89 | /** | 
| 90 | * For storage, get an ID for the pointer to the object you're going to | 90 | * For storage, get an ID for the pointer to the object you're going to | 
| diff --git a/src/archivebase.h b/src/archivebase.h index c871bb5..18591f0 100644 --- a/src/archivebase.h +++ b/src/archivebase.h | |||
| @@ -9,6 +9,7 @@ | |||
| 9 | #define BU_ARCHIVE_BASE_H | 9 | #define BU_ARCHIVE_BASE_H | 
| 10 | 10 | ||
| 11 | #include <stdint.h> | 11 | #include <stdint.h> | 
| 12 | #include <unistd.h> | ||
| 12 | 13 | ||
| 13 | namespace Bu | 14 | namespace Bu | 
| 14 | { | 15 | { | 
| @@ -19,8 +20,8 @@ namespace Bu | |||
| 19 | virtual ~ArchiveBase(); | 20 | virtual ~ArchiveBase(); | 
| 20 | 21 | ||
| 21 | virtual void close()=0; | 22 | virtual void close()=0; | 
| 22 | virtual void write( const void *pData, int32_t iLength )=0; | 23 | virtual void write( const void *pData, size_t iLength )=0; | 
| 23 | virtual void read( void *pData, int32_t iLength )=0; | 24 | virtual void read( void *pData, size_t iLength )=0; | 
| 24 | virtual bool isLoading()=0; | 25 | virtual bool isLoading()=0; | 
| 25 | }; | 26 | }; | 
| 26 | 27 | ||
| diff --git a/src/array.h b/src/array.h index eeee677..f225c97 100644 --- a/src/array.h +++ b/src/array.h | |||
| @@ -18,9 +18,18 @@ namespace Bu | |||
| 18 | subExceptionDecl( ArrayException ) | 18 | subExceptionDecl( ArrayException ) | 
| 19 | 19 | ||
| 20 | template<typename value, int inc, typename valuealloc> | 20 | template<typename value, int inc, typename valuealloc> | 
| 21 | class Array; | ||
| 22 | |||
| 23 | /** @cond DEVEL */ | ||
| 24 | template<typename value, int inc, typename valuealloc> | ||
| 21 | class ArrayCore | 25 | class ArrayCore | 
| 22 | { | 26 | { | 
| 23 | public: | 27 | friend class Array<value, inc, valuealloc>; | 
| 28 | friend class SharedCore< | ||
| 29 | Array<value, inc, valuealloc>, | ||
| 30 | ArrayCore<value, inc, valuealloc> | ||
| 31 | >; | ||
| 32 | private: | ||
| 24 | ArrayCore() : | 33 | ArrayCore() : | 
| 25 | pData( NULL ), | 34 | pData( NULL ), | 
| 26 | iSize( 0 ), | 35 | iSize( 0 ), | 
| @@ -99,6 +108,7 @@ namespace Bu | |||
| 99 | long iSize; | 108 | long iSize; | 
| 100 | long iCapacity; | 109 | long iCapacity; | 
| 101 | }; | 110 | }; | 
| 111 | /** @endcond */ | ||
| 102 | 112 | ||
| 103 | /** | 113 | /** | 
| 104 | * Array type container, just like a normal array only flexible and keeps | 114 | * Array type container, just like a normal array only flexible and keeps | 
| @@ -110,16 +120,20 @@ namespace Bu | |||
| 110 | *@ingroup Containers | 120 | *@ingroup Containers | 
| 111 | */ | 121 | */ | 
| 112 | template<typename value, int inc=10, typename valuealloc=std::allocator<value> > | 122 | template<typename value, int inc=10, typename valuealloc=std::allocator<value> > | 
| 113 | class Array : public SharedCore<ArrayCore<value, inc, valuealloc> > | 123 | class Array : public SharedCore< | 
| 124 | Array<value, inc, valuealloc>, | ||
| 125 | ArrayCore<value, inc, valuealloc> | ||
| 126 | > | ||
| 114 | { | 127 | { | 
| 115 | private: | 128 | private: | 
| 116 | typedef class Array<value, inc, valuealloc> MyType; | 129 | typedef class Array<value, inc, valuealloc> MyType; | 
| 117 | typedef class ArrayCore<value, inc, valuealloc> Core; | 130 | typedef class ArrayCore<value, inc, valuealloc> Core; | 
| 118 | 131 | ||
| 119 | protected: | 132 | protected: | 
| 120 | using SharedCore< Core >::core; | 133 | using SharedCore<MyType, Core>::core; | 
| 121 | using SharedCore< Core >::_hardCopy; | 134 | using SharedCore<MyType, Core>::_hardCopy; | 
| 122 | using SharedCore< Core >::_allocateCore; | 135 | using SharedCore<MyType, Core>::_resetCore; | 
| 136 | using SharedCore<MyType, Core>::_allocateCore; | ||
| 123 | 137 | ||
| 124 | public: | 138 | public: | 
| 125 | struct const_iterator; | 139 | struct const_iterator; | 
| @@ -130,7 +144,7 @@ namespace Bu | |||
| 130 | } | 144 | } | 
| 131 | 145 | ||
| 132 | Array( const MyType &src ) : | 146 | Array( const MyType &src ) : | 
| 133 | SharedCore< Core >( src ) | 147 | SharedCore<MyType, Core >( src ) | 
| 134 | { | 148 | { | 
| 135 | } | 149 | } | 
| 136 | 150 | ||
| @@ -168,8 +182,7 @@ namespace Bu | |||
| 168 | */ | 182 | */ | 
| 169 | void clear() | 183 | void clear() | 
| 170 | { | 184 | { | 
| 171 | _hardCopy(); | 185 | _resetCore(); | 
| 172 | core->clear(); | ||
| 173 | } | 186 | } | 
| 174 | 187 | ||
| 175 | MyType &append( const value &rVal ) | 188 | MyType &append( const value &rVal ) | 
| diff --git a/src/cachestorefiles.h b/src/cachestorefiles.h index 1906b13..c2cf091 100644 --- a/src/cachestorefiles.h +++ b/src/cachestorefiles.h | |||
| @@ -83,7 +83,7 @@ namespace Bu | |||
| 83 | } | 83 | } | 
| 84 | } | 84 | } | 
| 85 | 85 | ||
| 86 | virtual void unload( obtype *pObj, const keytype &key ) | 86 | virtual void unload( obtype *pObj, const keytype & ) | 
| 87 | { | 87 | { | 
| 88 | delete pObj; | 88 | delete pObj; | 
| 89 | } | 89 | } | 
| diff --git a/src/cachestoremyriad.h b/src/cachestoremyriad.h index b90793d..21c84e6 100644 --- a/src/cachestoremyriad.h +++ b/src/cachestoremyriad.h | |||
| @@ -85,7 +85,7 @@ namespace Bu | |||
| 85 | return pOb; | 85 | return pOb; | 
| 86 | } | 86 | } | 
| 87 | 87 | ||
| 88 | virtual void unload( obtype *pObj, const keytype &key ) | 88 | virtual void unload( obtype *pObj, const keytype & ) | 
| 89 | { | 89 | { | 
| 90 | delete pObj; | 90 | delete pObj; | 
| 91 | } | 91 | } | 
| diff --git a/src/client.cpp b/src/client.cpp index becd1bd..b635c8b 100644 --- a/src/client.cpp +++ b/src/client.cpp | |||
| @@ -6,7 +6,7 @@ | |||
| 6 | */ | 6 | */ | 
| 7 | 7 | ||
| 8 | #include "bu/client.h" | 8 | #include "bu/client.h" | 
| 9 | #include "bu/socket.h" | 9 | #include "bu/tcpsocket.h" | 
| 10 | #include <stdlib.h> | 10 | #include <stdlib.h> | 
| 11 | #include <errno.h> | 11 | #include <errno.h> | 
| 12 | #include "bu/protocol.h" | 12 | #include "bu/protocol.h" | 
| @@ -16,7 +16,8 @@ | |||
| 16 | /** Read buffer size. */ | 16 | /** Read buffer size. */ | 
| 17 | #define RBS (1024*2) | 17 | #define RBS (1024*2) | 
| 18 | 18 | ||
| 19 | Bu::Client::Client( Bu::Socket *pSocket, class Bu::ClientLinkFactory *pfLink ) : | 19 | Bu::Client::Client( Bu::TcpSocket *pSocket, | 
| 20 | class Bu::ClientLinkFactory *pfLink ) : | ||
| 20 | pTopStream( pSocket ), | 21 | pTopStream( pSocket ), | 
| 21 | pSocket( pSocket ), | 22 | pSocket( pSocket ), | 
| 22 | pProto( NULL ), | 23 | pProto( NULL ), | 
| @@ -59,7 +60,7 @@ void Bu::Client::processInput() | |||
| 59 | break; | 60 | break; | 
| 60 | } | 61 | } | 
| 61 | } | 62 | } | 
| 62 | catch( Bu::SocketException &e ) | 63 | catch( Bu::TcpSocketException &e ) | 
| 63 | { | 64 | { | 
| 64 | pTopStream->close(); | 65 | pTopStream->close(); | 
| 65 | bWantsDisconnect = true; | 66 | bWantsDisconnect = true; | 
| @@ -195,7 +196,7 @@ long Bu::Client::getOutputSize() | |||
| 195 | return qbWrite.getSize(); | 196 | return qbWrite.getSize(); | 
| 196 | } | 197 | } | 
| 197 | 198 | ||
| 198 | const Bu::Socket *Bu::Client::getSocket() const | 199 | const Bu::TcpSocket *Bu::Client::getSocket() const | 
| 199 | { | 200 | { | 
| 200 | return pSocket; | 201 | return pSocket; | 
| 201 | } | 202 | } | 
| diff --git a/src/client.h b/src/client.h index f336524..096df2f 100644 --- a/src/client.h +++ b/src/client.h | |||
| @@ -17,7 +17,7 @@ namespace Bu | |||
| 17 | { | 17 | { | 
| 18 | class Protocol; | 18 | class Protocol; | 
| 19 | class Stream; | 19 | class Stream; | 
| 20 | class Socket; | 20 | class TcpSocket; | 
| 21 | class ClientLinkFactory; | 21 | class ClientLinkFactory; | 
| 22 | 22 | ||
| 23 | /** | 23 | /** | 
| @@ -26,7 +26,7 @@ namespace Bu | |||
| 26 | class Client : public Bu::Stream | 26 | class Client : public Bu::Stream | 
| 27 | { | 27 | { | 
| 28 | public: | 28 | public: | 
| 29 | Client( Bu::Socket *pSocket, Bu::ClientLinkFactory *pfLink ); | 29 | Client( Bu::TcpSocket *pSocket, Bu::ClientLinkFactory *pfLink ); | 
| 30 | virtual ~Client(); | 30 | virtual ~Client(); | 
| 31 | 31 | ||
| 32 | void processInput(); | 32 | void processInput(); | 
| @@ -58,7 +58,7 @@ namespace Bu | |||
| 58 | void close(); | 58 | void close(); | 
| 59 | void tick(); | 59 | void tick(); | 
| 60 | 60 | ||
| 61 | const Bu::Socket *getSocket() const; | 61 | const Bu::TcpSocket *getSocket() const; | 
| 62 | 62 | ||
| 63 | void disconnect(); | 63 | void disconnect(); | 
| 64 | bool wantsDisconnect(); | 64 | bool wantsDisconnect(); | 
| @@ -117,7 +117,7 @@ namespace Bu | |||
| 117 | typedef Bu::List<Bu::Stream *> FilterList; | 117 | typedef Bu::List<Bu::Stream *> FilterList; | 
| 118 | FilterList lFilts; | 118 | FilterList lFilts; | 
| 119 | Bu::Stream *pTopStream; | 119 | Bu::Stream *pTopStream; | 
| 120 | Bu::Socket *pSocket; | 120 | Bu::TcpSocket *pSocket; | 
| 121 | Bu::Protocol *pProto; | 121 | Bu::Protocol *pProto; | 
| 122 | Bu::QueueBuf qbRead; | 122 | Bu::QueueBuf qbRead; | 
| 123 | Bu::QueueBuf qbWrite; | 123 | Bu::QueueBuf qbWrite; | 
| diff --git a/src/fastcgi.cpp b/src/fastcgi.cpp index 8168928..ca3010e 100644 --- a/src/fastcgi.cpp +++ b/src/fastcgi.cpp | |||
| @@ -24,14 +24,14 @@ Bu::FastCgi::FastCgi() : | |||
| 24 | pSrv( NULL ), | 24 | pSrv( NULL ), | 
| 25 | bRunning( true ) | 25 | bRunning( true ) | 
| 26 | { | 26 | { | 
| 27 | pSrv = new Bu::ServerSocket( STDIN_FILENO, false ); | 27 | pSrv = new Bu::TcpServerSocket( STDIN_FILENO, false ); | 
| 28 | } | 28 | } | 
| 29 | 29 | ||
| 30 | Bu::FastCgi::FastCgi( int iPort ) : | 30 | Bu::FastCgi::FastCgi( int iPort ) : | 
| 31 | pSrv( NULL ), | 31 | pSrv( NULL ), | 
| 32 | bRunning( true ) | 32 | bRunning( true ) | 
| 33 | { | 33 | { | 
| 34 | pSrv = new Bu::ServerSocket( iPort ); | 34 | pSrv = new Bu::TcpServerSocket( iPort ); | 
| 35 | } | 35 | } | 
| 36 | 36 | ||
| 37 | Bu::FastCgi::~FastCgi() | 37 | Bu::FastCgi::~FastCgi() | 
| @@ -64,17 +64,17 @@ bool Bu::FastCgi::isEmbedded() | |||
| 64 | #endif | 64 | #endif | 
| 65 | } | 65 | } | 
| 66 | 66 | ||
| 67 | void Bu::FastCgi::read( Bu::Socket &s, Bu::FastCgi::Record &r ) | 67 | void Bu::FastCgi::read( Bu::TcpSocket &s, Bu::FastCgi::Record &r ) | 
| 68 | { | 68 | { | 
| 69 | int iRead = s.read( &r, sizeof(Record) ); | 69 | int iRead = s.read( &r, sizeof(Record) ); | 
| 70 | if( iRead != sizeof(Record) ) | 70 | if( iRead != sizeof(Record) ) | 
| 71 | throw Bu::SocketException("Hey, the size %d is wrong for Record. (%s)", | 71 | throw Bu::TcpSocketException("Hey, the size %d is wrong for Record. (%s)", | 
| 72 | iRead, strerror( errno ) ); | 72 | iRead, strerror( errno ) ); | 
| 73 | r.uRequestId = ntohs( r.uRequestId ); | 73 | r.uRequestId = ntohs( r.uRequestId ); | 
| 74 | r.uContentLength = ntohs( r.uContentLength ); | 74 | r.uContentLength = ntohs( r.uContentLength ); | 
| 75 | } | 75 | } | 
| 76 | 76 | ||
| 77 | void Bu::FastCgi::write( Bu::Socket &s, Bu::FastCgi::Record r ) | 77 | void Bu::FastCgi::write( Bu::TcpSocket &s, Bu::FastCgi::Record r ) | 
| 78 | { | 78 | { | 
| 79 | // sio << "Out -> " << r << sio.nl; | 79 | // sio << "Out -> " << r << sio.nl; | 
| 80 | r.uRequestId = htons( r.uRequestId ); | 80 | r.uRequestId = htons( r.uRequestId ); | 
| @@ -82,19 +82,19 @@ void Bu::FastCgi::write( Bu::Socket &s, Bu::FastCgi::Record r ) | |||
| 82 | s.write( &r, sizeof(Record) ); | 82 | s.write( &r, sizeof(Record) ); | 
| 83 | } | 83 | } | 
| 84 | 84 | ||
| 85 | void Bu::FastCgi::read( Bu::Socket &s, Bu::FastCgi::BeginRequestBody &b ) | 85 | void Bu::FastCgi::read( Bu::TcpSocket &s, Bu::FastCgi::BeginRequestBody &b ) | 
| 86 | { | 86 | { | 
| 87 | s.read( &b, sizeof(BeginRequestBody) ); | 87 | s.read( &b, sizeof(BeginRequestBody) ); | 
| 88 | b.uRole = ntohs( b.uRole ); | 88 | b.uRole = ntohs( b.uRole ); | 
| 89 | } | 89 | } | 
| 90 | 90 | ||
| 91 | void Bu::FastCgi::write( Bu::Socket &s, Bu::FastCgi::EndRequestBody b ) | 91 | void Bu::FastCgi::write( Bu::TcpSocket &s, Bu::FastCgi::EndRequestBody b ) | 
| 92 | { | 92 | { | 
| 93 | b.uStatus = htonl( b.uStatus ); | 93 | b.uStatus = htonl( b.uStatus ); | 
| 94 | s.write( &b, sizeof(b) ); | 94 | s.write( &b, sizeof(b) ); | 
| 95 | } | 95 | } | 
| 96 | 96 | ||
| 97 | uint32_t Bu::FastCgi::readLen( Bu::Socket &s, uint16_t &uRead ) | 97 | uint32_t Bu::FastCgi::readLen( Bu::TcpSocket &s, uint16_t &uRead ) | 
| 98 | { | 98 | { | 
| 99 | uint8_t uByte[4]; | 99 | uint8_t uByte[4]; | 
| 100 | s.read( uByte, 1 ); | 100 | s.read( uByte, 1 ); | 
| @@ -107,7 +107,7 @@ uint32_t Bu::FastCgi::readLen( Bu::Socket &s, uint16_t &uRead ) | |||
| 107 | return ((uByte[0]&0x7f)<<24)|(uByte[1]<<16)|(uByte[2]<<8)|(uByte[3]); | 107 | return ((uByte[0]&0x7f)<<24)|(uByte[1]<<16)|(uByte[2]<<8)|(uByte[3]); | 
| 108 | } | 108 | } | 
| 109 | 109 | ||
| 110 | void Bu::FastCgi::readPair( Bu::Socket &s, StrHash &hParams, uint16_t &uRead ) | 110 | void Bu::FastCgi::readPair( Bu::TcpSocket &s, StrHash &hParams, uint16_t &uRead ) | 
| 111 | { | 111 | { | 
| 112 | uint32_t uName = readLen( s, uRead ); | 112 | uint32_t uName = readLen( s, uRead ); | 
| 113 | uint32_t uValue = readLen( s, uRead ); | 113 | uint32_t uValue = readLen( s, uRead ); | 
| @@ -162,7 +162,7 @@ void Bu::FastCgi::run() | |||
| 162 | if( iSock < 0 ) | 162 | if( iSock < 0 ) | 
| 163 | continue; | 163 | continue; | 
| 164 | 164 | ||
| 165 | Bu::Socket s( iSock ); | 165 | Bu::TcpSocket s( iSock ); | 
| 166 | s.setBlocking( true ); | 166 | s.setBlocking( true ); | 
| 167 | // sio << "Got connection, blocking? " << s.isBlocking() << sio.nl; | 167 | // sio << "Got connection, blocking? " << s.isBlocking() << sio.nl; | 
| 168 | try | 168 | try | 
| @@ -362,7 +362,7 @@ void Bu::FastCgi::run() | |||
| 362 | } | 362 | } | 
| 363 | } | 363 | } | 
| 364 | } | 364 | } | 
| 365 | catch( Bu::SocketException &e ) | 365 | catch( Bu::TcpSocketException &e ) | 
| 366 | { | 366 | { | 
| 367 | // sio << "Bu::SocketException: " << e.what() << sio.nl << | 367 | // sio << "Bu::SocketException: " << e.what() << sio.nl << | 
| 368 | // "\tSocket open: " << s.isOpen() << sio.nl; | 368 | // "\tSocket open: " << s.isOpen() << sio.nl; | 
| diff --git a/src/fastcgi.h b/src/fastcgi.h index 262872c..7c1c04c 100644 --- a/src/fastcgi.h +++ b/src/fastcgi.h | |||
| @@ -11,8 +11,8 @@ | |||
| 11 | #include "bu/fstring.h" | 11 | #include "bu/fstring.h" | 
| 12 | #include "bu/hash.h" | 12 | #include "bu/hash.h" | 
| 13 | #include "bu/array.h" | 13 | #include "bu/array.h" | 
| 14 | #include "bu/socket.h" | 14 | #include "bu/tcpsocket.h" | 
| 15 | #include "bu/serversocket.h" | 15 | #include "bu/tcpserversocket.h" | 
| 16 | 16 | ||
| 17 | namespace Bu | 17 | namespace Bu | 
| 18 | { | 18 | { | 
| @@ -109,18 +109,18 @@ namespace Bu | |||
| 109 | virtual void onUninit() { }; | 109 | virtual void onUninit() { }; | 
| 110 | 110 | ||
| 111 | private: | 111 | private: | 
| 112 | void read( Bu::Socket &s, Record &r ); | 112 | void read( Bu::TcpSocket &s, Record &r ); | 
| 113 | void read( Bu::Socket &s, BeginRequestBody &b ); | 113 | void read( Bu::TcpSocket &s, BeginRequestBody &b ); | 
| 114 | uint32_t readLen( Bu::Socket &s, uint16_t &uUsed ); | 114 | uint32_t readLen( Bu::TcpSocket &s, uint16_t &uUsed ); | 
| 115 | void readPair( Bu::Socket &s, StrHash &hParams, uint16_t &uUsed ); | 115 | void readPair( Bu::TcpSocket &s, StrHash &hParams, uint16_t &uUsed ); | 
| 116 | 116 | ||
| 117 | void write( Bu::Socket &s, Record r ); | 117 | void write( Bu::TcpSocket &s, Record r ); | 
| 118 | void write( Bu::Socket &s, EndRequestBody b ); | 118 | void write( Bu::TcpSocket &s, EndRequestBody b ); | 
| 119 | 119 | ||
| 120 | bool hasChannel( int iChan ); | 120 | bool hasChannel( int iChan ); | 
| 121 | 121 | ||
| 122 | private: | 122 | private: | 
| 123 | Bu::ServerSocket *pSrv; | 123 | Bu::TcpServerSocket *pSrv; | 
| 124 | bool bRunning; | 124 | bool bRunning; | 
| 125 | Bu::Array<Channel *> aChannel; | 125 | Bu::Array<Channel *> aChannel; | 
| 126 | }; | 126 | }; | 
| diff --git a/src/fbasicstring.h b/src/fbasicstring.h index 4c33479..064ff16 100644 --- a/src/fbasicstring.h +++ b/src/fbasicstring.h | |||
| @@ -25,6 +25,7 @@ | |||
| 25 | 25 | ||
| 26 | namespace Bu | 26 | namespace Bu | 
| 27 | { | 27 | { | 
| 28 | /** @cond DEVEL */ | ||
| 28 | template< typename chr > | 29 | template< typename chr > | 
| 29 | struct FStringChunk | 30 | struct FStringChunk | 
| 30 | { | 31 | { | 
| @@ -34,10 +35,55 @@ namespace Bu | |||
| 34 | }; | 35 | }; | 
| 35 | 36 | ||
| 36 | #define cpy( dest, src, size ) Bu::memcpy( dest, src, size*sizeof(chr) ) | 37 | #define cpy( dest, src, size ) Bu::memcpy( dest, src, size*sizeof(chr) ) | 
| 38 | |||
| 39 | template< typename chr, int nMinSize, typename chralloc, | ||
| 40 | typename chunkalloc> class FBasicString; | ||
| 41 | |||
| 42 | template<typename chr> | ||
| 43 | size_t strlen( const chr *pData ) | ||
| 44 | { | ||
| 45 | for( size_t tLen = 0;; ++tLen ) | ||
| 46 | { | ||
| 47 | if( pData[tLen] == (chr)0 ) | ||
| 48 | return tLen; | ||
| 49 | } | ||
| 50 | return -1; | ||
| 51 | } | ||
| 52 | |||
| 53 | template<char> | ||
| 54 | size_t strlen( const char *pData ) | ||
| 55 | { | ||
| 56 | return ::strlen( pData ); | ||
| 57 | } | ||
| 58 | |||
| 59 | template<typename chr> | ||
| 60 | int strncmp( const chr *a, const chr *b, size_t iLen ) | ||
| 61 | { | ||
| 62 | for( size_t iPos = 0; iPos < iLen; iPos++ ) | ||
| 63 | { | ||
| 64 | if( a[iPos] != b[iPos] ) | ||
| 65 | { | ||
| 66 | return a[iPos]-b[iPos]; | ||
| 67 | } | ||
| 68 | } | ||
| 69 | return 0; | ||
| 70 | } | ||
| 71 | |||
| 72 | template<char> | ||
| 73 | int strncmp( const char *a, const char *b, size_t iLen ) | ||
| 74 | { | ||
| 75 | return ::strncmp( a, b, iLen ); | ||
| 76 | } | ||
| 37 | 77 | ||
| 38 | template<typename chr, int nMinSize, typename chralloc, typename chunkalloc> | 78 | template<typename chr, int nMinSize, typename chralloc, typename chunkalloc> | 
| 39 | struct FStringCore | 79 | struct FStringCore | 
| 40 | { | 80 | { | 
| 81 | friend class FBasicString<chr, nMinSize, chralloc, chunkalloc>; | ||
| 82 | friend class SharedCore< | ||
| 83 | FBasicString<chr, nMinSize, chralloc, chunkalloc>, | ||
| 84 | FStringCore<chr, nMinSize, chralloc, chunkalloc> | ||
| 85 | >; | ||
| 86 | private: | ||
| 41 | typedef struct FStringCore<chr, nMinSize, chralloc, chunkalloc> MyType; | 87 | typedef struct FStringCore<chr, nMinSize, chralloc, chunkalloc> MyType; | 
| 42 | typedef struct FStringChunk<chr> Chunk; | 88 | typedef struct FStringChunk<chr> Chunk; | 
| 43 | FStringCore() : | 89 | FStringCore() : | 
| @@ -157,6 +203,7 @@ namespace Bu | |||
| 157 | nLength += pNewChunk->nLength; | 203 | nLength += pNewChunk->nLength; | 
| 158 | } | 204 | } | 
| 159 | }; | 205 | }; | 
| 206 | /** @endcond */ | ||
| 160 | 207 | ||
| 161 | /** | 208 | /** | 
| 162 | * Flexible String class. This class was designed with string passing and | 209 | * Flexible String class. This class was designed with string passing and | 
| @@ -174,55 +221,24 @@ namespace Bu | |||
| 174 | *@param chralloc (typename) Memory Allocator for chr | 221 | *@param chralloc (typename) Memory Allocator for chr | 
| 175 | *@param chunkalloc (typename) Memory Allocator for chr chunks | 222 | *@param chunkalloc (typename) Memory Allocator for chr chunks | 
| 176 | */ | 223 | */ | 
| 177 | template< typename chr, int nMinSize=256, typename chralloc=std::allocator<chr>, typename chunkalloc=std::allocator<struct FStringChunk<chr> > > | 224 | template< typename chr, int nMinSize=256, | 
| 178 | class FBasicString : public SharedCore< FStringCore<chr, nMinSize, chralloc, chunkalloc> > | 225 | typename chralloc=std::allocator<chr>, | 
| 226 | typename chunkalloc=std::allocator<struct FStringChunk<chr> > > | ||
| 227 | class FBasicString : public SharedCore< | ||
| 228 | FBasicString<chr, nMinSize, chralloc, chunkalloc>, | ||
| 229 | FStringCore<chr, nMinSize, chralloc, chunkalloc> > | ||
| 179 | { | 230 | { | 
| 180 | protected: | 231 | protected: | 
| 181 | typedef struct FStringChunk<chr> Chunk; | 232 | typedef struct FStringChunk<chr> Chunk; | 
| 182 | typedef struct FBasicString<chr, nMinSize, chralloc, chunkalloc> MyType; | 233 | typedef struct FBasicString<chr, nMinSize, chralloc, chunkalloc> MyType; | 
| 183 | typedef struct FStringCore<chr, nMinSize, chralloc, chunkalloc> Core; | 234 | typedef struct FStringCore<chr, nMinSize, chralloc, chunkalloc> Core; | 
| 184 | 235 | ||
| 185 | using SharedCore< Core >::core; | 236 | using SharedCore<MyType, Core >::core; | 
| 186 | using SharedCore< Core >::_hardCopy; | 237 | using SharedCore<MyType, Core >::_hardCopy; | 
| 187 | |||
| 188 | public: | ||
| 189 | FBasicString() | ||
| 190 | { | ||
| 191 | } | ||
| 192 | |||
| 193 | FBasicString( const chr *pData ) | ||
| 194 | { | ||
| 195 | append( pData ); | ||
| 196 | } | ||
| 197 | |||
| 198 | FBasicString( const chr *pData, long nLength ) | ||
| 199 | { | ||
| 200 | append( pData, nLength ); | ||
| 201 | } | ||
| 202 | 238 | ||
| 203 | FBasicString( const MyType &rSrc ) : | 239 | public: // Iterators | 
| 204 | SharedCore<Core>( rSrc ) | ||
| 205 | { | ||
| 206 | } | ||
| 207 | |||
| 208 | FBasicString( const MyType &rSrc, long nLength ) | ||
| 209 | { | ||
| 210 | append( rSrc, nLength ); | ||
| 211 | } | ||
| 212 | |||
| 213 | FBasicString( const MyType &rSrc, long nStart, long nLength ) | ||
| 214 | { | ||
| 215 | append( rSrc, nStart, nLength ); | ||
| 216 | } | ||
| 217 | |||
| 218 | FBasicString( long nSize ) | ||
| 219 | { | ||
| 220 | core->pFirst = core->pLast = core->newChunk( nSize ); | ||
| 221 | core->nLength = nSize; | ||
| 222 | } | ||
| 223 | |||
| 224 | struct iterator; | 240 | struct iterator; | 
| 225 | struct const_iterator | 241 | typedef struct const_iterator | 
| 226 | { | 242 | { | 
| 227 | friend class FBasicString<chr, nMinSize, chralloc, chunkalloc>; | 243 | friend class FBasicString<chr, nMinSize, chralloc, chunkalloc>; | 
| 228 | friend struct iterator; | 244 | friend struct iterator; | 
| @@ -243,7 +259,7 @@ namespace Bu | |||
| 243 | { | 259 | { | 
| 244 | } | 260 | } | 
| 245 | 261 | ||
| 246 | const_iterator( const iterator &i ) : | 262 | const_iterator( const struct iterator &i ) : | 
| 247 | pChunk( i.pChunk ), | 263 | pChunk( i.pChunk ), | 
| 248 | iPos( i.iPos ) | 264 | iPos( i.iPos ) | 
| 249 | { | 265 | { | 
| @@ -454,7 +470,7 @@ namespace Bu | |||
| 454 | } | 470 | } | 
| 455 | return const_iterator( NULL, 0 ); | 471 | return const_iterator( NULL, 0 ); | 
| 456 | } | 472 | } | 
| 457 | }; | 473 | } const_iterator; | 
| 458 | 474 | ||
| 459 | typedef struct iterator | 475 | typedef struct iterator | 
| 460 | { | 476 | { | 
| @@ -702,11 +718,41 @@ namespace Bu | |||
| 702 | } | 718 | } | 
| 703 | } iterator; | 719 | } iterator; | 
| 704 | 720 | ||
| 705 | typedef struct const_iterator const_iterator; | 721 | public: | 
| 722 | FBasicString() | ||
| 723 | { | ||
| 724 | } | ||
| 725 | |||
| 726 | FBasicString( const chr *pData ) | ||
| 727 | { | ||
| 728 | append( pData ); | ||
| 729 | } | ||
| 730 | |||
| 731 | FBasicString( const chr *pData, long nLength ) | ||
| 732 | { | ||
| 733 | append( pData, nLength ); | ||
| 734 | } | ||
| 735 | |||
| 736 | FBasicString( const MyType &rSrc ) : | ||
| 737 | SharedCore<MyType, Core>( rSrc ) | ||
| 738 | { | ||
| 739 | } | ||
| 706 | 740 | ||
| 707 | //typedef chr *iterator; | 741 | FBasicString( const MyType &rSrc, long nLength ) | 
| 708 | // typedef const chr *const_iterator; | 742 | { | 
| 709 | // typedef iterator const_iterator; | 743 | append( rSrc, nLength ); | 
| 744 | } | ||
| 745 | |||
| 746 | FBasicString( const MyType &rSrc, long nStart, long nLength ) | ||
| 747 | { | ||
| 748 | append( rSrc, nStart, nLength ); | ||
| 749 | } | ||
| 750 | |||
| 751 | FBasicString( long nSize ) | ||
| 752 | { | ||
| 753 | core->pFirst = core->pLast = core->newChunk( nSize ); | ||
| 754 | core->nLength = nSize; | ||
| 755 | } | ||
| 710 | 756 | ||
| 711 | FBasicString( const const_iterator &s ) | 757 | FBasicString( const const_iterator &s ) | 
| 712 | { | 758 | { | 
| @@ -731,14 +777,8 @@ namespace Bu | |||
| 731 | if( !pData ) return; | 777 | if( !pData ) return; | 
| 732 | long nLen; | 778 | long nLen; | 
| 733 | for( nLen = 0; pData[nLen] != (chr)0; nLen++ ) { } | 779 | for( nLen = 0; pData[nLen] != (chr)0; nLen++ ) { } | 
| 734 | if( nLen == 0 ) | ||
| 735 | return; | ||
| 736 | |||
| 737 | Chunk *pNew = core->newChunk( nLen ); | ||
| 738 | cpy( pNew->pData, pData, nLen ); | ||
| 739 | 780 | ||
| 740 | _hardCopy(); | 781 | append( pData, 0, nLen ); | 
| 741 | core->appendChunk( pNew ); | ||
| 742 | } | 782 | } | 
| 743 | 783 | ||
| 744 | /** | 784 | /** | 
| @@ -748,15 +788,7 @@ namespace Bu | |||
| 748 | */ | 788 | */ | 
| 749 | void append( const chr *pData, long nLen ) | 789 | void append( const chr *pData, long nLen ) | 
| 750 | { | 790 | { | 
| 751 | if( nLen == 0 ) | 791 | append( pData, 0, nLen ); | 
| 752 | return; | ||
| 753 | |||
| 754 | Chunk *pNew = core->newChunk( nLen ); | ||
| 755 | |||
| 756 | cpy( pNew->pData, pData, nLen ); | ||
| 757 | |||
| 758 | _hardCopy(); | ||
| 759 | core->appendChunk( pNew ); | ||
| 760 | } | 792 | } | 
| 761 | 793 | ||
| 762 | /** | 794 | /** | 
| @@ -767,15 +799,37 @@ namespace Bu | |||
| 767 | */ | 799 | */ | 
| 768 | void append( const chr *pData, long nStart, long nLen ) | 800 | void append( const chr *pData, long nStart, long nLen ) | 
| 769 | { | 801 | { | 
| 770 | if( nLen == 0 ) | 802 | if( !pData ) return; | 
| 803 | if( nLen <= 0 ) | ||
| 771 | return; | 804 | return; | 
| 772 | 805 | ||
| 773 | Chunk *pNew = core->newChunk( nLen ); | 806 | pData += nStart; | 
| 774 | |||
| 775 | cpy( pNew->pData, pData+nStart, nLen ); | ||
| 776 | 807 | ||
| 777 | _hardCopy(); | 808 | _hardCopy(); | 
| 778 | core->appendChunk( pNew ); | 809 | |
| 810 | if( core->pLast && core->pLast->nLength < nMinSize ) | ||
| 811 | { | ||
| 812 | int nAmnt = nMinSize - core->pLast->nLength; | ||
| 813 | if( nAmnt > nLen ) | ||
| 814 | nAmnt = nLen; | ||
| 815 | cpy( | ||
| 816 | core->pLast->pData+core->pLast->nLength, | ||
| 817 | pData, | ||
| 818 | nAmnt | ||
| 819 | ); | ||
| 820 | pData += nAmnt; | ||
| 821 | core->pLast->nLength += nAmnt; | ||
| 822 | nLen -= nAmnt; | ||
| 823 | core->nLength += nAmnt; | ||
| 824 | } | ||
| 825 | |||
| 826 | if( nLen > 0 ) | ||
| 827 | { | ||
| 828 | Chunk *pNew = core->newChunk( nLen ); | ||
| 829 | cpy( pNew->pData, pData, nLen ); | ||
| 830 | core->appendChunk( pNew ); | ||
| 831 | // core->nLength += nLen; | ||
| 832 | } | ||
| 779 | } | 833 | } | 
| 780 | 834 | ||
| 781 | /** | 835 | /** | 
| @@ -804,7 +858,7 @@ namespace Bu | |||
| 804 | */ | 858 | */ | 
| 805 | void append( const MyType & sData ) | 859 | void append( const MyType & sData ) | 
| 806 | { | 860 | { | 
| 807 | append( sData.getStr(), sData.getSize() ); | 861 | append( sData.getStr(), 0, sData.getSize() ); | 
| 808 | } | 862 | } | 
| 809 | 863 | ||
| 810 | /** | 864 | /** | 
| @@ -815,7 +869,7 @@ namespace Bu | |||
| 815 | */ | 869 | */ | 
| 816 | void append( const MyType & sData, long nLen ) | 870 | void append( const MyType & sData, long nLen ) | 
| 817 | { | 871 | { | 
| 818 | append( sData.getStr(), nLen ); | 872 | append( sData.getStr(), 0, nLen ); | 
| 819 | } | 873 | } | 
| 820 | 874 | ||
| 821 | /** | 875 | /** | 
| @@ -1029,7 +1083,7 @@ namespace Bu | |||
| 1029 | */ | 1083 | */ | 
| 1030 | void insert( long nPos, const chr *pData ) | 1084 | void insert( long nPos, const chr *pData ) | 
| 1031 | { | 1085 | { | 
| 1032 | insert( nPos, pData, strlen( pData ) ); | 1086 | insert( nPos, pData, Bu::strlen( pData ) ); | 
| 1033 | } | 1087 | } | 
| 1034 | 1088 | ||
| 1035 | void remove( long nPos, long nLen ) | 1089 | void remove( long nPos, long nLen ) | 
| @@ -1155,13 +1209,13 @@ namespace Bu | |||
| 1155 | if( iStart < 0 ) | 1209 | if( iStart < 0 ) | 
| 1156 | iStart = 0; | 1210 | iStart = 0; | 
| 1157 | if( iStart >= core->nLength ) | 1211 | if( iStart >= core->nLength ) | 
| 1158 | return ""; | 1212 | return (const chr[]){(chr)0}; | 
| 1159 | if( iSize < 0 ) | 1213 | if( iSize < 0 ) | 
| 1160 | iSize = core->nLength; | 1214 | iSize = core->nLength; | 
| 1161 | if( iStart+iSize > core->nLength ) | 1215 | if( iStart+iSize > core->nLength ) | 
| 1162 | iSize = core->nLength-iStart; | 1216 | iSize = core->nLength-iStart; | 
| 1163 | if( iSize == 0 ) | 1217 | if( iSize == 0 ) | 
| 1164 | return ""; | 1218 | return (const chr[]){(chr)0}; | 
| 1165 | 1219 | ||
| 1166 | flatten(); | 1220 | flatten(); | 
| 1167 | MyType ret( core->pFirst->pData+iStart, iSize ); | 1221 | MyType ret( core->pFirst->pData+iStart, iSize ); | 
| @@ -1212,37 +1266,6 @@ namespace Bu | |||
| 1212 | } | 1266 | } | 
| 1213 | } | 1267 | } | 
| 1214 | 1268 | ||
| 1215 | /** | ||
| 1216 | * (std::string compatability) Get a pointer to the string array. | ||
| 1217 | *@returns (chr *) The string data. | ||
| 1218 | */ | ||
| 1219 | DEPRECATED | ||
| 1220 | chr *c_str() | ||
| 1221 | { | ||
| 1222 | if( core->pFirst == NULL || core->nLength == 0 ) | ||
| 1223 | return NULL; | ||
| 1224 | |||
| 1225 | flatten(); | ||
| 1226 | _hardCopy(); | ||
| 1227 | core->pFirst->pData[core->nLength] = (chr)0; | ||
| 1228 | return core->pFirst->pData; | ||
| 1229 | } | ||
| 1230 | |||
| 1231 | /** | ||
| 1232 | * (std::string compatability) Get a const pointer to the string array. | ||
| 1233 | *@returns (const chr *) The string data. | ||
| 1234 | */ | ||
| 1235 | DEPRECATED | ||
| 1236 | const chr *c_str() const | ||
| 1237 | { | ||
| 1238 | if( core->pFirst == NULL || core->nLength == 0 ) | ||
| 1239 | return NULL; | ||
| 1240 | |||
| 1241 | flatten(); | ||
| 1242 | core->pFirst->pData[core->nLength] = (chr)0; | ||
| 1243 | return core->pFirst->pData; | ||
| 1244 | } | ||
| 1245 | |||
| 1246 | Bu::List<MyType> split( const chr c ) const | 1269 | Bu::List<MyType> split( const chr c ) const | 
| 1247 | { | 1270 | { | 
| 1248 | Bu::List<MyType> ret; | 1271 | Bu::List<MyType> ret; | 
| @@ -1423,11 +1446,11 @@ namespace Bu | |||
| 1423 | wordexp_t result; | 1446 | wordexp_t result; | 
| 1424 | 1447 | ||
| 1425 | /* Expand the string for the program to run. */ | 1448 | /* Expand the string for the program to run. */ | 
| 1426 | switch (wordexp (core->pFirst->pData, &result, 0)) | 1449 | switch (wordexp ((char *)core->pFirst->pData, &result, 0)) | 
| 1427 | { | 1450 | { | 
| 1428 | case 0: /* Successful. */ | 1451 | case 0: /* Successful. */ | 
| 1429 | { | 1452 | { | 
| 1430 | set( result.we_wordv[0] ); | 1453 | set( (chr *)result.we_wordv[0] ); | 
| 1431 | wordfree( &result ); | 1454 | wordfree( &result ); | 
| 1432 | return; | 1455 | return; | 
| 1433 | } | 1456 | } | 
| @@ -1941,7 +1964,7 @@ namespace Bu | |||
| 1941 | long iLen = vsnprintf( NULL, 0, sFrmt, ap ); | 1964 | long iLen = vsnprintf( NULL, 0, sFrmt, ap ); | 
| 1942 | 1965 | ||
| 1943 | Chunk *pNew = core->newChunk( iLen ); | 1966 | Chunk *pNew = core->newChunk( iLen ); | 
| 1944 | vsnprintf( pNew->pData, iLen+1, sFrmt, ap ); | 1967 | vsnprintf( (char *)pNew->pData, iLen+1, sFrmt, ap ); | 
| 1945 | core->appendChunk( pNew ); | 1968 | core->appendChunk( pNew ); | 
| 1946 | 1969 | ||
| 1947 | va_end( ap ); | 1970 | va_end( ap ); | 
| @@ -1958,7 +1981,7 @@ namespace Bu | |||
| 1958 | long iLen = vsnprintf( NULL, 0, sFrmt, ap ); | 1981 | long iLen = vsnprintf( NULL, 0, sFrmt, ap ); | 
| 1959 | 1982 | ||
| 1960 | Chunk *pNew = core->newChunk( iLen ); | 1983 | Chunk *pNew = core->newChunk( iLen ); | 
| 1961 | vsnprintf( pNew->pData, iLen+1, sFrmt, ap ); | 1984 | vsnprintf( (char *)pNew->pData, iLen+1, sFrmt, ap ); | 
| 1962 | core->appendChunk( pNew ); | 1985 | core->appendChunk( pNew ); | 
| 1963 | 1986 | ||
| 1964 | va_end( ap ); | 1987 | va_end( ap ); | 
| @@ -1975,7 +1998,7 @@ namespace Bu | |||
| 1975 | long iLen = vsnprintf( NULL, 0, sFrmt, ap ); | 1998 | long iLen = vsnprintf( NULL, 0, sFrmt, ap ); | 
| 1976 | 1999 | ||
| 1977 | Chunk *pNew = core->newChunk( iLen ); | 2000 | Chunk *pNew = core->newChunk( iLen ); | 
| 1978 | vsnprintf( pNew->pData, iLen+1, sFrmt, ap ); | 2001 | vsnprintf( (char *)pNew->pData, iLen+1, sFrmt, ap ); | 
| 1979 | core->prependChunk( pNew ); | 2002 | core->prependChunk( pNew ); | 
| 1980 | 2003 | ||
| 1981 | va_end( ap ); | 2004 | va_end( ap ); | 
| diff --git a/src/file.cpp b/src/file.cpp index 0566ee8..008b88e 100644 --- a/src/file.cpp +++ b/src/file.cpp | |||
| @@ -11,7 +11,6 @@ | |||
| 11 | #include <sys/stat.h> | 11 | #include <sys/stat.h> | 
| 12 | #include <fcntl.h> | 12 | #include <fcntl.h> | 
| 13 | #include <unistd.h> | 13 | #include <unistd.h> | 
| 14 | #include <stdlib.h> // for mkstemp | ||
| 15 | #include <time.h> | 14 | #include <time.h> | 
| 16 | 15 | ||
| 17 | #include "bu/config.h" | 16 | #include "bu/config.h" | 
| @@ -12,7 +12,8 @@ | |||
| 12 | #include "bu/exceptionbase.h" | 12 | #include "bu/exceptionbase.h" | 
| 13 | #include "bu/list.h" | 13 | #include "bu/list.h" | 
| 14 | #include "bu/util.h" | 14 | #include "bu/util.h" | 
| 15 | #include "archivebase.h" | 15 | #include "bu/archivebase.h" | 
| 16 | #include "bu/sharedcore.h" | ||
| 16 | 17 | ||
| 17 | #define bitsToBytes( n ) (n/32+(n%32>0 ? 1 : 0)) | 18 | #define bitsToBytes( n ) (n/32+(n%32>0 ? 1 : 0)) | 
| 18 | 19 | ||
| @@ -49,138 +50,357 @@ namespace Bu | |||
| 49 | } | 50 | } | 
| 50 | }; | 51 | }; | 
| 51 | 52 | ||
| 52 | template<typename key, typename value, typename sizecalc = __calcNextTSize_fast, typename keyalloc = std::allocator<key>, typename valuealloc = std::allocator<value>, typename challoc = std::allocator<uint32_t> > | 53 | template<typename key, typename value, typename sizecalc, typename keyalloc, | 
| 54 | typename valuealloc, typename challoc> | ||
| 53 | class Hash; | 55 | class Hash; | 
| 54 | 56 | ||
| 55 | template< typename key, typename _value, typename sizecalc = __calcNextTSize_fast, typename keyalloc = std::allocator<key>, typename valuealloc = std::allocator<_value>, typename challoc = std::allocator<uint32_t> > | 57 | /** @cond DEVEL */ | 
| 56 | struct HashProxy | 58 | template<typename key, typename value, typename sizecalc, typename keyalloc, | 
| 59 | typename valuealloc, typename challoc > | ||
| 60 | class HashCore | ||
| 57 | { | 61 | { | 
| 58 | friend class Hash<key, _value, sizecalc, keyalloc, valuealloc, challoc>; | 62 | friend class Hash<key, value, sizecalc, keyalloc, valuealloc, challoc>; | 
| 63 | friend class SharedCore< | ||
| 64 | Hash<key, value, sizecalc, keyalloc, valuealloc, challoc>, | ||
| 65 | HashCore<key, value, sizecalc, keyalloc, valuealloc, challoc> | ||
| 66 | >; | ||
| 59 | private: | 67 | private: | 
| 60 | HashProxy( Hash<key, _value, sizecalc, keyalloc, valuealloc, challoc> &h, const key *k, uint32_t nPos, uint32_t hash ) : | 68 | HashCore() : | 
| 61 | hsh( h ), | 69 | nCapacity( 0 ), | 
| 62 | pKey( k ), | 70 | nFilled( 0 ), | 
| 63 | nPos( nPos ), | 71 | nDeleted( 0 ), | 
| 64 | hash( hash ), | 72 | bFilled( NULL ), | 
| 65 | bFilled( false ) | 73 | bDeleted( NULL ), | 
| 74 | aKeys( NULL ), | ||
| 75 | aValues( NULL ), | ||
| 76 | aHashCodes( NULL ) | ||
| 66 | { | 77 | { | 
| 67 | } | 78 | } | 
| 68 | 79 | ||
| 69 | HashProxy( Hash<key, _value, sizecalc, keyalloc, valuealloc, challoc> &h, uint32_t nPos, _value *pValue ) : | 80 | virtual ~HashCore() | 
| 70 | hsh( h ), | ||
| 71 | nPos( nPos ), | ||
| 72 | pValue( pValue ), | ||
| 73 | bFilled( true ) | ||
| 74 | { | 81 | { | 
| 82 | clear(); | ||
| 75 | } | 83 | } | 
| 76 | 84 | ||
| 77 | Hash<key, _value, sizecalc, keyalloc, valuealloc, challoc> &hsh; | 85 | void init() | 
| 78 | const key *pKey; | 86 | { | 
| 79 | uint32_t nPos; | 87 | if( nCapacity > 0 ) | 
| 80 | _value *pValue; | 88 | return; | 
| 81 | uint32_t hash; | ||
| 82 | bool bFilled; | ||
| 83 | 89 | ||
| 84 | public: | 90 | nCapacity = 11; | 
| 85 | /** | 91 | nKeysSize = bitsToBytes( nCapacity ); | 
| 86 | * Cast operator for HashProxy. | 92 | bFilled = ca.allocate( nKeysSize ); | 
| 87 | *@returns (value_type &) The value the HashProxy is pointing to. | 93 | bDeleted = ca.allocate( nKeysSize ); | 
| 88 | */ | 94 | clearBits(); | 
| 89 | operator _value &() | 95 | |
| 96 | aHashCodes = ca.allocate( nCapacity ); | ||
| 97 | aKeys = ka.allocate( nCapacity ); | ||
| 98 | aValues = va.allocate( nCapacity ); | ||
| 99 | } | ||
| 100 | |||
| 101 | void clearBits() | ||
| 90 | { | 102 | { | 
| 91 | if( bFilled == false ) | 103 | if( nCapacity == 0 ) | 
| 92 | throw HashException( | 104 | return; | 
| 93 | excodeNotFilled, | 105 | |
| 94 | "No data assosiated with that key." | 106 | for( uint32_t j = 0; j < nKeysSize; j++ ) | 
| 95 | ); | 107 | { | 
| 96 | return *pValue; | 108 | bFilled[j] = bDeleted[j] = 0; | 
| 109 | } | ||
| 97 | } | 110 | } | 
| 98 | 111 | ||
| 99 | /** | 112 | void fill( uint32_t loc, const key &k, const value &v, uint32_t hash ) | 
| 100 | * Direct function for retrieving a value out of the HashProxy. | ||
| 101 | *@returns (value_type &) The value pointed to by this HashProxy. | ||
| 102 | */ | ||
| 103 | DEPRECATED | ||
| 104 | _value &value() | ||
| 105 | { | 113 | { | 
| 106 | if( bFilled == false ) | 114 | init(); | 
| 107 | throw HashException( | 115 | |
| 108 | excodeNotFilled, | 116 | bFilled[loc/32] |= (1<<(loc%32)); | 
| 109 | "No data assosiated with that key." | 117 | va.construct( &aValues[loc], v ); | 
| 110 | ); | 118 | ka.construct( &aKeys[loc], k ); | 
| 111 | return *pValue; | 119 | aHashCodes[loc] = hash; | 
| 120 | nFilled++; | ||
| 121 | //printf("Filled: %d, Deleted: %d, Capacity: %d\n", | ||
| 122 | // nFilled, nDeleted, nCapacity ); | ||
| 123 | } | ||
| 124 | |||
| 125 | void _erase( uint32_t loc ) | ||
| 126 | { | ||
| 127 | if( nCapacity == 0 ) | ||
| 128 | return; | ||
| 129 | |||
| 130 | bDeleted[loc/32] |= (1<<(loc%32)); | ||
| 131 | va.destroy( &aValues[loc] ); | ||
| 132 | ka.destroy( &aKeys[loc] ); | ||
| 133 | nDeleted++; | ||
| 134 | //printf("Filled: %d, Deleted: %d, Capacity: %d\n", | ||
| 135 | // nFilled, nDeleted, nCapacity ); | ||
| 136 | } | ||
| 137 | |||
| 138 | key &getKeyAtPos( uint32_t nPos ) | ||
| 139 | { | ||
| 140 | if( nPos >= nCapacity ) | ||
| 141 | throw HashException("Referenced position invalid."); | ||
| 142 | return aKeys[nPos]; | ||
| 112 | } | 143 | } | 
| 113 | 144 | ||
| 114 | /** | 145 | const key &getKeyAtPos( uint32_t nPos ) const | 
| 115 | * Direct function for retrieving a value out of the HashProxy. | ||
| 116 | *@returns (value_type &) The value pointed to by this HashProxy. | ||
| 117 | */ | ||
| 118 | _value &getValue() | ||
| 119 | { | 146 | { | 
| 120 | if( bFilled == false ) | 147 | if( nPos >= nCapacity ) | 
| 121 | throw HashException( | 148 | throw HashException("Referenced position invalid."); | 
| 122 | excodeNotFilled, | 149 | return aKeys[nPos]; | 
| 123 | "No data assosiated with that key." | 150 | } | 
| 124 | ); | 151 | |
| 125 | return *pValue; | 152 | value &getValueAtPos( uint32_t nPos ) | 
| 153 | { | ||
| 154 | if( nPos >= nCapacity ) | ||
| 155 | throw HashException("Referenced position invalid."); | ||
| 156 | return aValues[nPos]; | ||
| 157 | } | ||
| 158 | |||
| 159 | const value &getValueAtPos( uint32_t nPos ) const | ||
| 160 | { | ||
| 161 | if( nPos >= nCapacity ) | ||
| 162 | throw HashException("Referenced position invalid."); | ||
| 163 | return aValues[nPos]; | ||
| 126 | } | 164 | } | 
| 127 | 165 | ||
| 128 | /** | 166 | uint32_t getFirstPos( bool &bFinished ) const | 
| 129 | * Whether this HashProxy points to something real or not. | 167 | { | 
| 130 | */ | 168 | for( uint32_t j = 0; j < nCapacity; j++ ) | 
| 131 | bool isFilled() | 169 | { | 
| 170 | if( isFilled( j ) ) | ||
| 171 | if( !isDeleted( j ) ) | ||
| 172 | return j; | ||
| 173 | } | ||
| 174 | |||
| 175 | bFinished = true; | ||
| 176 | return 0; | ||
| 177 | } | ||
| 178 | |||
| 179 | uint32_t getNextPos( uint32_t nPos, bool &bFinished ) const | ||
| 132 | { | 180 | { | 
| 133 | return bFilled; | 181 | for( uint32_t j = nPos+1; j < nCapacity; j++ ) | 
| 182 | { | ||
| 183 | if( isFilled( j ) ) | ||
| 184 | if( !isDeleted( j ) ) | ||
| 185 | return j; | ||
| 186 | } | ||
| 187 | |||
| 188 | bFinished = true; | ||
| 189 | return 0; | ||
| 134 | } | 190 | } | 
| 135 | 191 | ||
| 136 | /** | 192 | uint32_t probe( uint32_t hash, const key &k, bool &bFill, bool rehash=true ) | 
| 137 | * Erase the data pointed to by this HashProxy. | ||
| 138 | */ | ||
| 139 | void erase() | ||
| 140 | { | 193 | { | 
| 141 | if( bFilled ) | 194 | init(); | 
| 195 | |||
| 196 | uint32_t nCur = hash%nCapacity; | ||
| 197 | |||
| 198 | // First we scan to see if the key is already there, abort if we | ||
| 199 | // run out of probing room, or we find a non-filled entry | ||
| 200 | int8_t j; | ||
| 201 | for( j = 0; | ||
| 202 | isFilled( nCur ) && j < 32; | ||
| 203 | nCur = (nCur + (1<<j))%nCapacity, j++ | ||
| 204 | ) | ||
| 205 | { | ||
| 206 | // Is this the same hash code we were looking for? | ||
| 207 | if( hash == aHashCodes[nCur] ) | ||
| 208 | { | ||
| 209 | // Skip over deleted entries. Deleted entries are also filled, | ||
| 210 | // so we only have to do this check here. | ||
| 211 | if( isDeleted( nCur ) ) | ||
| 212 | continue; | ||
| 213 | |||
| 214 | // Is it really the same key? (for safety) | ||
| 215 | if( __cmpHashKeys( aKeys[nCur], k ) == true ) | ||
| 216 | { | ||
| 217 | bFill = true; | ||
| 218 | return nCur; | ||
| 219 | } | ||
| 220 | } | ||
| 221 | } | ||
| 222 | |||
| 223 | // This is our insurance, if the table is full, then go ahead and | ||
| 224 | // rehash, then try again. | ||
| 225 | if( (isFilled( nCur ) || j == 32) && rehash == true ) | ||
| 142 | { | 226 | { | 
| 143 | hsh._erase( nPos ); | 227 | reHash( szCalc( nCapacity, nFilled, nDeleted ) ); | 
| 144 | hsh.onDelete(); | 228 | |
| 229 | // This is potentially dangerous, and could cause an infinite loop. | ||
| 230 | // Be careful writing probe, eh? | ||
| 231 | return probe( hash, k, bFill ); | ||
| 145 | } | 232 | } | 
| 233 | |||
| 234 | bFill = false; | ||
| 235 | return nCur; | ||
| 146 | } | 236 | } | 
| 237 | |||
| 238 | uint32_t probe( uint32_t hash, key k, bool &bFill ) const | ||
| 239 | { | ||
| 240 | if( nCapacity == 0 ) | ||
| 241 | throw Bu::ExceptionBase("Probe in empty hash table."); | ||
| 147 | 242 | ||
| 148 | /** | 243 | uint32_t nCur = hash%nCapacity; | 
| 149 | * Assign data to this point in the hash table. | 244 | |
| 150 | *@param nval (value_type) the data to assign. | 245 | // First we scan to see if the key is already there, abort if we | 
| 151 | */ | 246 | // run out of probing room, or we find a non-filled entry | 
| 152 | _value operator=( _value nval ) | 247 | for( int8_t j = 0; | 
| 248 | isFilled( nCur ) && j < 32; | ||
| 249 | nCur = (nCur + (1<<j))%nCapacity, j++ | ||
| 250 | ) | ||
| 251 | { | ||
| 252 | // Is this the same hash code we were looking for? | ||
| 253 | if( hash == aHashCodes[nCur] ) | ||
| 254 | { | ||
| 255 | // Skip over deleted entries. Deleted entries are also filled, | ||
| 256 | // so we only have to do this check here. | ||
| 257 | if( isDeleted( nCur ) ) | ||
| 258 | continue; | ||
| 259 | |||
| 260 | // Is it really the same key? (for safety) | ||
| 261 | if( __cmpHashKeys( aKeys[nCur], k ) == true ) | ||
| 262 | { | ||
| 263 | bFill = true; | ||
| 264 | return nCur; | ||
| 265 | } | ||
| 266 | } | ||
| 267 | } | ||
| 268 | |||
| 269 | bFill = false; | ||
| 270 | return nCur; | ||
| 271 | } | ||
| 272 | |||
| 273 | void insert( const key &k, const value &v ) | ||
| 153 | { | 274 | { | 
| 154 | if( bFilled ) | 275 | uint32_t hash = __calcHashCode( k ); | 
| 276 | bool bFill; | ||
| 277 | uint32_t nPos = probe( hash, k, bFill ); | ||
| 278 | |||
| 279 | if( bFill ) | ||
| 155 | { | 280 | { | 
| 156 | hsh.va.destroy( &hsh.aValues[nPos] ); | 281 | va.destroy( &aValues[nPos] ); | 
| 157 | hsh.va.construct( &hsh.aValues[nPos], nval ); | 282 | va.construct( &aValues[nPos], v ); | 
| 158 | hsh.onUpdate(); | ||
| 159 | } | 283 | } | 
| 160 | else | 284 | else | 
| 161 | { | 285 | { | 
| 162 | hsh.fill( nPos, *pKey, nval, hash ); | 286 | fill( nPos, k, v, hash ); | 
| 163 | hsh.onInsert(); | 287 | } | 
| 288 | } | ||
| 289 | |||
| 290 | void reHash( uint32_t nNewSize ) | ||
| 291 | { | ||
| 292 | //printf("---REHASH---"); | ||
| 293 | //printf("Filled: %d, Deleted: %d, Capacity: %d\n", | ||
| 294 | // nFilled, nDeleted, nCapacity ); | ||
| 295 | |||
| 296 | // Save all the old data | ||
| 297 | uint32_t nOldCapacity = nCapacity; | ||
| 298 | uint32_t *bOldFilled = bFilled; | ||
| 299 | uint32_t *aOldHashCodes = aHashCodes; | ||
| 300 | uint32_t nOldKeysSize = nKeysSize; | ||
| 301 | uint32_t *bOldDeleted = bDeleted; | ||
| 302 | value *aOldValues = aValues; | ||
| 303 | key *aOldKeys = aKeys; | ||
| 304 | |||
| 305 | // Calculate new sizes | ||
| 306 | nCapacity = nNewSize; | ||
| 307 | nKeysSize = bitsToBytes( nCapacity ); | ||
| 308 | |||
| 309 | // Allocate new memory + prep | ||
| 310 | bFilled = ca.allocate( nKeysSize ); | ||
| 311 | bDeleted = ca.allocate( nKeysSize ); | ||
| 312 | clearBits(); | ||
| 313 | |||
| 314 | aHashCodes = ca.allocate( nCapacity ); | ||
| 315 | aKeys = ka.allocate( nCapacity ); | ||
| 316 | aValues = va.allocate( nCapacity ); | ||
| 317 | |||
| 318 | nDeleted = nFilled = 0; | ||
| 319 | |||
| 320 | // Re-insert all of the old data (except deleted items) | ||
| 321 | for( uint32_t j = 0; j < nOldCapacity; j++ ) | ||
| 322 | { | ||
| 323 | if( (bOldFilled[j/32]&(1<<(j%32)))!=0 && | ||
| 324 | (bOldDeleted[j/32]&(1<<(j%32)))==0 ) | ||
| 325 | { | ||
| 326 | insert( aOldKeys[j], aOldValues[j] ); | ||
| 327 | } | ||
| 328 | } | ||
| 329 | |||
| 330 | // Delete all of the old data | ||
| 331 | for( uint32_t j = 0; j < nOldCapacity; j++ ) | ||
| 332 | { | ||
| 333 | if( (bOldFilled[j/32]&(1<<(j%32)))!=0 && | ||
| 334 | (bOldDeleted[j/32]&(1<<(j%32)))==0 ) | ||
| 335 | { | ||
| 336 | va.destroy( &aOldValues[j] ); | ||
| 337 | ka.destroy( &aOldKeys[j] ); | ||
| 338 | } | ||
| 164 | } | 339 | } | 
| 340 | va.deallocate( aOldValues, nOldCapacity ); | ||
| 341 | ka.deallocate( aOldKeys, nOldCapacity ); | ||
| 342 | ca.deallocate( bOldFilled, nOldKeysSize ); | ||
| 343 | ca.deallocate( bOldDeleted, nOldKeysSize ); | ||
| 344 | ca.deallocate( aOldHashCodes, nOldCapacity ); | ||
| 345 | } | ||
| 165 | 346 | ||
| 166 | return nval; | 347 | bool isFilled( uint32_t loc ) const | 
| 348 | { | ||
| 349 | if( loc >= nCapacity ) | ||
| 350 | throw HashException("Referenced position invalid."); | ||
| 351 | return (bFilled[loc/32]&(1<<(loc%32)))!=0; | ||
| 167 | } | 352 | } | 
| 168 | 353 | ||
| 169 | /** | 354 | bool isDeleted( uint32_t loc ) const | 
| 170 | * Pointer extraction operator. Access to members of data pointed to | ||
| 171 | * by HashProxy. | ||
| 172 | *@returns (value_type *) | ||
| 173 | */ | ||
| 174 | _value *operator->() | ||
| 175 | { | 355 | { | 
| 176 | if( bFilled == false ) | 356 | if( loc >= nCapacity ) | 
| 177 | throw HashException( | 357 | throw HashException("Referenced position invalid."); | 
| 178 | excodeNotFilled, | 358 | return (bDeleted[loc/32]&(1<<(loc%32)))!=0; | 
| 179 | "No data assosiated with that key." | 359 | } | 
| 180 | ); | 360 | |
| 181 | return pValue; | 361 | void clear() | 
| 362 | { | ||
| 363 | for( uint32_t j = 0; j < nCapacity; j++ ) | ||
| 364 | { | ||
| 365 | if( isFilled( j ) ) | ||
| 366 | if( !isDeleted( j ) ) | ||
| 367 | { | ||
| 368 | va.destroy( &aValues[j] ); | ||
| 369 | ka.destroy( &aKeys[j] ); | ||
| 370 | } | ||
| 371 | } | ||
| 372 | va.deallocate( aValues, nCapacity ); | ||
| 373 | ka.deallocate( aKeys, nCapacity ); | ||
| 374 | ca.deallocate( bFilled, nKeysSize ); | ||
| 375 | ca.deallocate( bDeleted, nKeysSize ); | ||
| 376 | ca.deallocate( aHashCodes, nCapacity ); | ||
| 377 | |||
| 378 | bFilled = NULL; | ||
| 379 | bDeleted = NULL; | ||
| 380 | aKeys = NULL; | ||
| 381 | aValues = NULL; | ||
| 382 | aHashCodes = NULL; | ||
| 383 | |||
| 384 | nCapacity = 0; | ||
| 385 | nFilled = 0; | ||
| 386 | nDeleted = 0; | ||
| 182 | } | 387 | } | 
| 388 | |||
| 389 | uint32_t nCapacity; | ||
| 390 | uint32_t nFilled; | ||
| 391 | uint32_t nDeleted; | ||
| 392 | uint32_t *bFilled; | ||
| 393 | uint32_t *bDeleted; | ||
| 394 | uint32_t nKeysSize; | ||
| 395 | key *aKeys; | ||
| 396 | value *aValues; | ||
| 397 | uint32_t *aHashCodes; | ||
| 398 | valuealloc va; | ||
| 399 | keyalloc ka; | ||
| 400 | challoc ca; | ||
| 401 | sizecalc szCalc; | ||
| 183 | }; | 402 | }; | 
| 403 | /** @endcond */ | ||
| 184 | 404 | ||
| 185 | /** | 405 | /** | 
| 186 | * Libbu++ Template Hash Table. This is your average hash table, that uses | 406 | * Libbu++ Template Hash Table. This is your average hash table, that uses | 
| @@ -224,119 +444,38 @@ namespace Bu | |||
| 224 | *@param challoc (typename) Byte allocator for bitflags | 444 | *@param challoc (typename) Byte allocator for bitflags | 
| 225 | *@ingroup Containers | 445 | *@ingroup Containers | 
| 226 | */ | 446 | */ | 
| 227 | template<typename key, typename value, typename sizecalc, typename keyalloc, typename valuealloc, typename challoc > | 447 | template<typename key, typename value, | 
| 228 | class Hash | 448 | typename sizecalc = __calcNextTSize_fast, | 
| 449 | typename keyalloc = std::allocator<key>, | ||
| 450 | typename valuealloc = std::allocator<value>, | ||
| 451 | typename challoc = std::allocator<uint32_t> | ||
| 452 | > | ||
| 453 | class Hash : public SharedCore< | ||
| 454 | Hash<key, value, sizecalc, keyalloc, valuealloc, challoc>, | ||
| 455 | HashCore<key, value, sizecalc, keyalloc, valuealloc, challoc> | ||
| 456 | > | ||
| 229 | { | 457 | { | 
| 230 | friend struct HashProxy<key, value, sizecalc, keyalloc, valuealloc, challoc>; | 458 | private: | 
| 459 | typedef class HashCore<key, value, sizecalc, keyalloc, valuealloc, challoc> Core; | ||
| 231 | typedef class Hash<key, value, sizecalc, keyalloc, valuealloc, challoc> MyType; | 460 | typedef class Hash<key, value, sizecalc, keyalloc, valuealloc, challoc> MyType; | 
| 461 | protected: | ||
| 462 | using SharedCore<MyType, Core>::core; | ||
| 463 | using SharedCore<MyType, Core>::_hardCopy; | ||
| 464 | using SharedCore<MyType, Core>::_resetCore; | ||
| 465 | using SharedCore<MyType, Core>::_allocateCore; | ||
| 466 | |||
| 232 | public: | 467 | public: | 
| 233 | Hash() : | 468 | Hash() | 
| 234 | nCapacity( 11 ), | ||
| 235 | nFilled( 0 ), | ||
| 236 | nDeleted( 0 ), | ||
| 237 | bFilled( NULL ), | ||
| 238 | bDeleted( NULL ), | ||
| 239 | aKeys( NULL ), | ||
| 240 | aValues( NULL ), | ||
| 241 | aHashCodes( NULL ) | ||
| 242 | { | ||
| 243 | nKeysSize = bitsToBytes( nCapacity ); | ||
| 244 | bFilled = ca.allocate( nKeysSize ); | ||
| 245 | bDeleted = ca.allocate( nKeysSize ); | ||
| 246 | clearBits(); | ||
| 247 | |||
| 248 | aHashCodes = ca.allocate( nCapacity ); | ||
| 249 | aKeys = ka.allocate( nCapacity ); | ||
| 250 | aValues = va.allocate( nCapacity ); | ||
| 251 | } | ||
| 252 | |||
| 253 | Hash( const Hash &src ) : | ||
| 254 | nCapacity( src.nCapacity ), | ||
| 255 | nFilled( 0 ), | ||
| 256 | nDeleted( 0 ), | ||
| 257 | bFilled( NULL ), | ||
| 258 | bDeleted( NULL ), | ||
| 259 | aKeys( NULL ), | ||
| 260 | aValues( NULL ), | ||
| 261 | aHashCodes( NULL ) | ||
| 262 | { | 469 | { | 
| 263 | nKeysSize = bitsToBytes( nCapacity ); | ||
| 264 | bFilled = ca.allocate( nKeysSize ); | ||
| 265 | bDeleted = ca.allocate( nKeysSize ); | ||
| 266 | clearBits(); | ||
| 267 | |||
| 268 | aHashCodes = ca.allocate( nCapacity ); | ||
| 269 | aKeys = ka.allocate( nCapacity ); | ||
| 270 | aValues = va.allocate( nCapacity ); | ||
| 271 | |||
| 272 | for( uint32_t j = 0; j < src.nCapacity; j++ ) | ||
| 273 | { | ||
| 274 | if( src.isFilled( j ) && !src.isDeleted( j ) ) | ||
| 275 | { | ||
| 276 | insert( src.aKeys[j], src.aValues[j] ); | ||
| 277 | } | ||
| 278 | } | ||
| 279 | } | 470 | } | 
| 280 | 471 | ||
| 281 | /** | 472 | Hash( const MyType &src ) : | 
| 282 | * Hashtable assignment operator. Clears this hashtable and | 473 | SharedCore<MyType, Core >( src ) | 
| 283 | * copies RH into it. | ||
| 284 | */ | ||
| 285 | Hash &operator=( const Hash &src ) | ||
| 286 | { | 474 | { | 
| 287 | for( uint32_t j = 0; j < nCapacity; j++ ) | ||
| 288 | { | ||
| 289 | if( isFilled( j ) && !isDeleted( j ) ) | ||
| 290 | { | ||
| 291 | va.destroy( &aValues[j] ); | ||
| 292 | ka.destroy( &aKeys[j] ); | ||
| 293 | } | ||
| 294 | } | ||
| 295 | va.deallocate( aValues, nCapacity ); | ||
| 296 | ka.deallocate( aKeys, nCapacity ); | ||
| 297 | ca.deallocate( bFilled, nKeysSize ); | ||
| 298 | ca.deallocate( bDeleted, nKeysSize ); | ||
| 299 | ca.deallocate( aHashCodes, nCapacity ); | ||
| 300 | |||
| 301 | nFilled = 0; | ||
| 302 | nDeleted = 0; | ||
| 303 | nCapacity = src.nCapacity; | ||
| 304 | nKeysSize = bitsToBytes( nCapacity ); | ||
| 305 | bFilled = ca.allocate( nKeysSize ); | ||
| 306 | bDeleted = ca.allocate( nKeysSize ); | ||
| 307 | clearBits(); | ||
| 308 | |||
| 309 | aHashCodes = ca.allocate( nCapacity ); | ||
| 310 | aKeys = ka.allocate( nCapacity ); | ||
| 311 | aValues = va.allocate( nCapacity ); | ||
| 312 | |||
| 313 | for( uint32_t j = 0; j < src.nCapacity; j++ ) | ||
| 314 | { | ||
| 315 | if( src.isFilled( j ) && !src.isDeleted( j ) ) | ||
| 316 | { | ||
| 317 | insert( src.aKeys[j], src.aValues[j] ); | ||
| 318 | } | ||
| 319 | } | ||
| 320 | |||
| 321 | return *this; | ||
| 322 | } | 475 | } | 
| 323 | 476 | ||
| 324 | virtual ~Hash() | 477 | virtual ~Hash() | 
| 325 | { | 478 | { | 
| 326 | for( uint32_t j = 0; j < nCapacity; j++ ) | ||
| 327 | { | ||
| 328 | if( isFilled( j ) ) | ||
| 329 | if( !isDeleted( j ) ) | ||
| 330 | { | ||
| 331 | va.destroy( &aValues[j] ); | ||
| 332 | ka.destroy( &aKeys[j] ); | ||
| 333 | } | ||
| 334 | } | ||
| 335 | va.deallocate( aValues, nCapacity ); | ||
| 336 | ka.deallocate( aKeys, nCapacity ); | ||
| 337 | ca.deallocate( bFilled, nKeysSize ); | ||
| 338 | ca.deallocate( bDeleted, nKeysSize ); | ||
| 339 | ca.deallocate( aHashCodes, nCapacity ); | ||
| 340 | } | 479 | } | 
| 341 | 480 | ||
| 342 | /** | 481 | /** | 
| @@ -345,7 +484,7 @@ namespace Bu | |||
| 345 | */ | 484 | */ | 
| 346 | uint32_t getCapacity() const | 485 | uint32_t getCapacity() const | 
| 347 | { | 486 | { | 
| 348 | return nCapacity; | 487 | return core->nCapacity; | 
| 349 | } | 488 | } | 
| 350 | 489 | ||
| 351 | /** | 490 | /** | 
| @@ -355,7 +494,7 @@ namespace Bu | |||
| 355 | */ | 494 | */ | 
| 356 | uint32_t getFill() const | 495 | uint32_t getFill() const | 
| 357 | { | 496 | { | 
| 358 | return nFilled; | 497 | return core->nFilled; | 
| 359 | } | 498 | } | 
| 360 | 499 | ||
| 361 | /** | 500 | /** | 
| @@ -364,12 +503,12 @@ namespace Bu | |||
| 364 | */ | 503 | */ | 
| 365 | uint32_t getSize() const | 504 | uint32_t getSize() const | 
| 366 | { | 505 | { | 
| 367 | return nFilled-nDeleted; | 506 | return core->nFilled-core->nDeleted; | 
| 368 | } | 507 | } | 
| 369 | 508 | ||
| 370 | bool isEmpty() const | 509 | bool isEmpty() const | 
| 371 | { | 510 | { | 
| 372 | return (nFilled-nDeleted) == 0; | 511 | return (core->nFilled-core->nDeleted) == 0; | 
| 373 | } | 512 | } | 
| 374 | 513 | ||
| 375 | /** | 514 | /** | 
| @@ -379,27 +518,140 @@ namespace Bu | |||
| 379 | */ | 518 | */ | 
| 380 | uint32_t getDeleted() const | 519 | uint32_t getDeleted() const | 
| 381 | { | 520 | { | 
| 382 | return nDeleted; | 521 | return core->nDeleted; | 
| 383 | } | 522 | } | 
| 384 | 523 | ||
| 524 | struct HashProxy | ||
| 525 | { | ||
| 526 | friend class Hash<key, value, sizecalc, keyalloc, valuealloc, challoc>; | ||
| 527 | private: | ||
| 528 | HashProxy( MyType &h, const key *k, uint32_t nPos, uint32_t hash ) : | ||
| 529 | hsh( h ), | ||
| 530 | pKey( k ), | ||
| 531 | nPos( nPos ), | ||
| 532 | hash( hash ), | ||
| 533 | bFilled( false ) | ||
| 534 | { | ||
| 535 | } | ||
| 536 | |||
| 537 | HashProxy( MyType &h, uint32_t nPos, value *pValue ) : | ||
| 538 | hsh( h ), | ||
| 539 | nPos( nPos ), | ||
| 540 | pValue( pValue ), | ||
| 541 | bFilled( true ) | ||
| 542 | { | ||
| 543 | } | ||
| 544 | |||
| 545 | MyType &hsh; | ||
| 546 | const key *pKey; | ||
| 547 | uint32_t nPos; | ||
| 548 | value *pValue; | ||
| 549 | uint32_t hash; | ||
| 550 | bool bFilled; | ||
| 551 | |||
| 552 | public: | ||
| 553 | /** | ||
| 554 | * Cast operator for HashProxy. | ||
| 555 | *@returns (value_type &) The value the HashProxy is pointing to. | ||
| 556 | */ | ||
| 557 | operator value &() | ||
| 558 | { | ||
| 559 | if( bFilled == false ) | ||
| 560 | throw HashException( | ||
| 561 | excodeNotFilled, | ||
| 562 | "No data assosiated with that key." | ||
| 563 | ); | ||
| 564 | return *pValue; | ||
| 565 | } | ||
| 566 | |||
| 567 | /** | ||
| 568 | * Direct function for retrieving a value out of the HashProxy. | ||
| 569 | *@returns (value_type &) The value pointed to by this HashProxy. | ||
| 570 | */ | ||
| 571 | value &getValue() | ||
| 572 | { | ||
| 573 | if( bFilled == false ) | ||
| 574 | throw HashException( | ||
| 575 | excodeNotFilled, | ||
| 576 | "No data assosiated with that key." | ||
| 577 | ); | ||
| 578 | return *pValue; | ||
| 579 | } | ||
| 580 | |||
| 581 | /** | ||
| 582 | * Whether this HashProxy points to something real or not. | ||
| 583 | */ | ||
| 584 | bool isFilled() | ||
| 585 | { | ||
| 586 | return bFilled; | ||
| 587 | } | ||
| 588 | |||
| 589 | /** | ||
| 590 | * Erase the data pointed to by this HashProxy. | ||
| 591 | */ | ||
| 592 | void erase() | ||
| 593 | { | ||
| 594 | if( bFilled ) | ||
| 595 | { | ||
| 596 | hsh.core->_erase( nPos ); | ||
| 597 | } | ||
| 598 | } | ||
| 599 | |||
| 600 | /** | ||
| 601 | * Assign data to this point in the hash table. | ||
| 602 | *@param nval (value_type) the data to assign. | ||
| 603 | */ | ||
| 604 | value operator=( value nval ) | ||
| 605 | { | ||
| 606 | if( bFilled ) | ||
| 607 | { | ||
| 608 | hsh.core->va.destroy( &hsh.core->aValues[nPos] ); | ||
| 609 | hsh.core->va.construct( &hsh.core->aValues[nPos], nval ); | ||
| 610 | } | ||
| 611 | else | ||
| 612 | { | ||
| 613 | hsh.core->fill( nPos, *pKey, nval, hash ); | ||
| 614 | } | ||
| 615 | |||
| 616 | return nval; | ||
| 617 | } | ||
| 618 | |||
| 619 | /** | ||
| 620 | * Pointer extraction operator. Access to members of data pointed to | ||
| 621 | * by HashProxy. | ||
| 622 | *@returns (value_type *) | ||
| 623 | */ | ||
| 624 | value *operator->() | ||
| 625 | { | ||
| 626 | if( bFilled == false ) | ||
| 627 | throw HashException( | ||
| 628 | excodeNotFilled, | ||
| 629 | "No data assosiated with that key." | ||
| 630 | ); | ||
| 631 | return pValue; | ||
| 632 | } | ||
| 633 | }; | ||
| 634 | |||
| 385 | /** | 635 | /** | 
| 386 | * Hash table index operator | 636 | * Hash table index operator | 
| 387 | *@param k (key_type) Key of data to be retrieved. | 637 | *@param k (key_type) Key of data to be retrieved. | 
| 388 | *@returns (HashProxy) Proxy pointing to the data. | 638 | *@returns (HashProxy) Proxy pointing to the data. | 
| 389 | */ | 639 | */ | 
| 390 | virtual HashProxy<key, value, sizecalc, keyalloc, valuealloc, challoc> operator[]( const key &k ) | 640 | HashProxy operator[]( const key &k ) | 
| 391 | { | 641 | { | 
| 642 | _hardCopy(); | ||
| 643 | |||
| 392 | uint32_t hash = __calcHashCode( k ); | 644 | uint32_t hash = __calcHashCode( k ); | 
| 393 | bool bFill; | 645 | bool bFill; | 
| 394 | uint32_t nPos = probe( hash, k, bFill ); | 646 | uint32_t nPos = core->probe( hash, k, bFill ); | 
| 395 | 647 | ||
| 396 | if( bFill ) | 648 | if( bFill ) | 
| 397 | { | 649 | { | 
| 398 | return HashProxy<key, value, sizecalc, keyalloc, valuealloc, challoc>( *this, nPos, &aValues[nPos] ); | 650 | return HashProxy( *this, nPos, &core->aValues[nPos] ); | 
| 399 | } | 651 | } | 
| 400 | else | 652 | else | 
| 401 | { | 653 | { | 
| 402 | return HashProxy<key, value, sizecalc, keyalloc, valuealloc, challoc>( *this, &k, nPos, hash ); | 654 | return HashProxy( *this, &k, nPos, hash ); | 
| 403 | } | 655 | } | 
| 404 | } | 656 | } | 
| 405 | 657 | ||
| @@ -408,39 +660,28 @@ namespace Bu | |||
| 408 | *@param k (key_type) Key to list the value under. | 660 | *@param k (key_type) Key to list the value under. | 
| 409 | *@param v (value_type) Value to store in the hash table. | 661 | *@param v (value_type) Value to store in the hash table. | 
| 410 | */ | 662 | */ | 
| 411 | virtual void insert( const key &k, const value &v ) | 663 | void insert( const key &k, const value &v ) | 
| 412 | { | 664 | { | 
| 413 | uint32_t hash = __calcHashCode( k ); | 665 | _hardCopy(); | 
| 414 | bool bFill; | ||
| 415 | uint32_t nPos = probe( hash, k, bFill ); | ||
| 416 | 666 | ||
| 417 | if( bFill ) | 667 | core->insert( k, v ); | 
| 418 | { | ||
| 419 | va.destroy( &aValues[nPos] ); | ||
| 420 | va.construct( &aValues[nPos], v ); | ||
| 421 | onUpdate(); | ||
| 422 | } | ||
| 423 | else | ||
| 424 | { | ||
| 425 | fill( nPos, k, v, hash ); | ||
| 426 | onInsert(); | ||
| 427 | } | ||
| 428 | } | 668 | } | 
| 429 | 669 | ||
| 430 | /** | 670 | /** | 
| 431 | * Remove a value from the hash table. | 671 | * Remove a value from the hash table. | 
| 432 | *@param k (key_type) The data under this key will be erased. | 672 | *@param k (key_type) The data under this key will be erased. | 
| 433 | */ | 673 | */ | 
| 434 | virtual void erase( const key &k ) | 674 | void erase( const key &k ) | 
| 435 | { | 675 | { | 
| 676 | _hardCopy(); | ||
| 677 | |||
| 436 | uint32_t hash = __calcHashCode( k ); | 678 | uint32_t hash = __calcHashCode( k ); | 
| 437 | bool bFill; | 679 | bool bFill; | 
| 438 | uint32_t nPos = probe( hash, k, bFill ); | 680 | uint32_t nPos = core->probe( hash, k, bFill ); | 
| 439 | 681 | ||
| 440 | if( bFill ) | 682 | if( bFill ) | 
| 441 | { | 683 | { | 
| 442 | _erase( nPos ); | 684 | core->_erase( nPos ); | 
| 443 | onDelete(); | ||
| 444 | } | 685 | } | 
| 445 | } | 686 | } | 
| 446 | 687 | ||
| @@ -450,14 +691,16 @@ namespace Bu | |||
| 450 | * Remove a value from the hash pointed to from an iterator. | 691 | * Remove a value from the hash pointed to from an iterator. | 
| 451 | *@param i (iterator &) The data to be erased. | 692 | *@param i (iterator &) The data to be erased. | 
| 452 | */ | 693 | */ | 
| 453 | virtual void erase( struct iterator &i ) | 694 | void erase( struct iterator &i ) | 
| 454 | { | 695 | { | 
| 455 | if( this != i.hsh ) | 696 | if( this != i.hsh ) | 
| 456 | throw HashException("This iterator didn't come from this Hash."); | 697 | throw HashException("This iterator didn't come from this Hash."); | 
| 457 | if( isFilled( i.nPos ) && !isDeleted( i.nPos ) ) | 698 | |
| 699 | _hardCopy(); | ||
| 700 | |||
| 701 | if( core->isFilled( i.nPos ) && !core->isDeleted( i.nPos ) ) | ||
| 458 | { | 702 | { | 
| 459 | _erase( i.nPos ); | 703 | core->_erase( i.nPos ); | 
| 460 | onDelete(); | ||
| 461 | } | 704 | } | 
| 462 | } | 705 | } | 
| 463 | 706 | ||
| @@ -466,17 +709,7 @@ namespace Bu | |||
| 466 | */ | 709 | */ | 
| 467 | virtual void clear() | 710 | virtual void clear() | 
| 468 | { | 711 | { | 
| 469 | for( uint32_t j = 0; j < nCapacity; j++ ) | 712 | _resetCore(); | 
| 470 | { | ||
| 471 | if( isFilled( j ) ) | ||
| 472 | if( !isDeleted( j ) ) | ||
| 473 | { | ||
| 474 | _erase( j ); | ||
| 475 | onDelete(); | ||
| 476 | } | ||
| 477 | } | ||
| 478 | |||
| 479 | clearBits(); | ||
| 480 | } | 713 | } | 
| 481 | 714 | ||
| 482 | /** | 715 | /** | 
| @@ -484,15 +717,17 @@ namespace Bu | |||
| 484 | *@param k (key_type) Key pointing to the data to be retrieved. | 717 | *@param k (key_type) Key pointing to the data to be retrieved. | 
| 485 | *@returns (value_type &) The data pointed to by (k). | 718 | *@returns (value_type &) The data pointed to by (k). | 
| 486 | */ | 719 | */ | 
| 487 | virtual value &get( const key &k ) | 720 | value &get( const key &k ) | 
| 488 | { | 721 | { | 
| 722 | _hardCopy(); | ||
| 723 | |||
| 489 | uint32_t hash = __calcHashCode( k ); | 724 | uint32_t hash = __calcHashCode( k ); | 
| 490 | bool bFill; | 725 | bool bFill; | 
| 491 | uint32_t nPos = probe( hash, k, bFill, false ); | 726 | uint32_t nPos = core->probe( hash, k, bFill, false ); | 
| 492 | 727 | ||
| 493 | if( bFill ) | 728 | if( bFill ) | 
| 494 | { | 729 | { | 
| 495 | return aValues[nPos]; | 730 | return core->aValues[nPos]; | 
| 496 | } | 731 | } | 
| 497 | else | 732 | else | 
| 498 | { | 733 | { | 
| @@ -509,15 +744,15 @@ namespace Bu | |||
| 509 | *@returns (const value_type &) A const version of the data pointed | 744 | *@returns (const value_type &) A const version of the data pointed | 
| 510 | * to by (k). | 745 | * to by (k). | 
| 511 | */ | 746 | */ | 
| 512 | virtual const value &get( const key &k ) const | 747 | const value &get( const key &k ) const | 
| 513 | { | 748 | { | 
| 514 | uint32_t hash = __calcHashCode( k ); | 749 | uint32_t hash = __calcHashCode( k ); | 
| 515 | bool bFill; | 750 | bool bFill; | 
| 516 | uint32_t nPos = probe( hash, k, bFill ); | 751 | uint32_t nPos = core->probe( hash, k, bFill ); | 
| 517 | 752 | ||
| 518 | if( bFill ) | 753 | if( bFill ) | 
| 519 | { | 754 | { | 
| 520 | return aValues[nPos]; | 755 | return core->aValues[nPos]; | 
| 521 | } | 756 | } | 
| 522 | else | 757 | else | 
| 523 | { | 758 | { | 
| @@ -533,18 +768,10 @@ namespace Bu | |||
| 533 | *@param k (key_type) The key to check. | 768 | *@param k (key_type) The key to check. | 
| 534 | *@returns (bool) Whether there was an item in the hash under key (k). | 769 | *@returns (bool) Whether there was an item in the hash under key (k). | 
| 535 | */ | 770 | */ | 
| 536 | virtual bool has( const key &k ) | 771 | bool has( const key &k ) const | 
| 537 | { | 772 | { | 
| 538 | bool bFill; | 773 | bool bFill; | 
| 539 | probe( __calcHashCode( k ), k, bFill, false ); | 774 | core->probe( __calcHashCode( k ), k, bFill ); | 
| 540 | |||
| 541 | return bFill; | ||
| 542 | } | ||
| 543 | |||
| 544 | virtual bool has( const key &k ) const | ||
| 545 | { | ||
| 546 | bool bFill; | ||
| 547 | probe( __calcHashCode( k ), k, bFill ); | ||
| 548 | 775 | ||
| 549 | return bFill; | 776 | return bFill; | 
| 550 | } | 777 | } | 
| @@ -561,7 +788,7 @@ namespace Bu | |||
| 561 | nPos( 0 ), | 788 | nPos( 0 ), | 
| 562 | bFinished( false ) | 789 | bFinished( false ) | 
| 563 | { | 790 | { | 
| 564 | nPos = hsh->getFirstPos( bFinished ); | 791 | nPos = hsh->core->getFirstPos( bFinished ); | 
| 565 | } | 792 | } | 
| 566 | 793 | ||
| 567 | iterator( MyType *hsh, bool bDone ) : | 794 | iterator( MyType *hsh, bool bDone ) : | 
| @@ -590,11 +817,6 @@ namespace Bu | |||
| 590 | { | 817 | { | 
| 591 | } | 818 | } | 
| 592 | 819 | ||
| 593 | DEPRECATED bool isActive() const | ||
| 594 | { | ||
| 595 | return !bFinished; | ||
| 596 | } | ||
| 597 | |||
| 598 | bool isValid() const | 820 | bool isValid() const | 
| 599 | { | 821 | { | 
| 600 | return !bFinished; | 822 | return !bFinished; | 
| @@ -611,7 +833,7 @@ namespace Bu | |||
| 611 | iterator operator++( int ) | 833 | iterator operator++( int ) | 
| 612 | { | 834 | { | 
| 613 | if( bFinished == false ) | 835 | if( bFinished == false ) | 
| 614 | nPos = hsh->getNextPos( nPos, bFinished ); | 836 | nPos = hsh->core->getNextPos( nPos, bFinished ); | 
| 615 | 837 | ||
| 616 | return *this; | 838 | return *this; | 
| 617 | } | 839 | } | 
| @@ -622,7 +844,7 @@ namespace Bu | |||
| 622 | iterator operator++() | 844 | iterator operator++() | 
| 623 | { | 845 | { | 
| 624 | if( bFinished == false ) | 846 | if( bFinished == false ) | 
| 625 | nPos = hsh->getNextPos( nPos, bFinished ); | 847 | nPos = hsh->core->getNextPos( nPos, bFinished ); | 
| 626 | 848 | ||
| 627 | return *this; | 849 | return *this; | 
| 628 | } | 850 | } | 
| @@ -671,21 +893,22 @@ namespace Bu | |||
| 671 | */ | 893 | */ | 
| 672 | value &operator *() | 894 | value &operator *() | 
| 673 | { | 895 | { | 
| 674 | return hsh->getValueAtPos( nPos ); | 896 | hsh->_hardCopy(); | 
| 897 | return hsh->core->getValueAtPos( nPos ); | ||
| 675 | } | 898 | } | 
| 676 | 899 | ||
| 677 | const value &operator *() const | 900 | const value &operator *() const | 
| 678 | { | 901 | { | 
| 679 | return hsh->getValueAtPos( nPos ); | 902 | return hsh->core->getValueAtPos( nPos ); | 
| 680 | } | 903 | } | 
| 681 | 904 | ||
| 682 | /** | 905 | /** | 
| 683 | * Get the key behind this iterator. | 906 | * Get the key behind this iterator. | 
| 684 | *@returns (key_type &) The key behind this iterator. | 907 | *@returns (key_type &) The key behind this iterator. | 
| 685 | */ | 908 | */ | 
| 686 | key &getKey() | 909 | const key &getKey() const | 
| 687 | { | 910 | { | 
| 688 | return hsh->getKeyAtPos( nPos ); | 911 | return hsh->core->getKeyAtPos( nPos ); | 
| 689 | } | 912 | } | 
| 690 | 913 | ||
| 691 | /** | 914 | /** | 
| @@ -694,7 +917,17 @@ namespace Bu | |||
| 694 | */ | 917 | */ | 
| 695 | value &getValue() | 918 | value &getValue() | 
| 696 | { | 919 | { | 
| 697 | return hsh->getValueAtPos( nPos ); | 920 | hsh->_hardCopy(); | 
| 921 | return hsh->core->getValueAtPos( nPos ); | ||
| 922 | } | ||
| 923 | |||
| 924 | /** | ||
| 925 | * Get the value behind this iterator. | ||
| 926 | *@returns (value_type &) The value behind this iterator. | ||
| 927 | */ | ||
| 928 | const value &getValue() const | ||
| 929 | { | ||
| 930 | return hsh->core->getValueAtPos( nPos ); | ||
| 698 | } | 931 | } | 
| 699 | } iterator; | 932 | } iterator; | 
| 700 | 933 | ||
| @@ -710,7 +943,7 @@ namespace Bu | |||
| 710 | nPos( 0 ), | 943 | nPos( 0 ), | 
| 711 | bFinished( false ) | 944 | bFinished( false ) | 
| 712 | { | 945 | { | 
| 713 | nPos = hsh->getFirstPos( bFinished ); | 946 | nPos = hsh->core->getFirstPos( bFinished ); | 
| 714 | } | 947 | } | 
| 715 | 948 | ||
| 716 | const_iterator( const MyType *hsh, bool bDone ) : | 949 | const_iterator( const MyType *hsh, bool bDone ) : | 
| @@ -762,7 +995,7 @@ namespace Bu | |||
| 762 | const_iterator operator++( int ) | 995 | const_iterator operator++( int ) | 
| 763 | { | 996 | { | 
| 764 | if( bFinished == false ) | 997 | if( bFinished == false ) | 
| 765 | nPos = hsh->getNextPos( nPos, bFinished ); | 998 | nPos = hsh->core->getNextPos( nPos, bFinished ); | 
| 766 | 999 | ||
| 767 | return *this; | 1000 | return *this; | 
| 768 | } | 1001 | } | 
| @@ -773,7 +1006,7 @@ namespace Bu | |||
| 773 | const_iterator operator++() | 1006 | const_iterator operator++() | 
| 774 | { | 1007 | { | 
| 775 | if( bFinished == false ) | 1008 | if( bFinished == false ) | 
| 776 | nPos = hsh->getNextPos( nPos, bFinished ); | 1009 | nPos = hsh->core->getNextPos( nPos, bFinished ); | 
| 777 | 1010 | ||
| 778 | return *this; | 1011 | return *this; | 
| 779 | } | 1012 | } | 
| @@ -822,7 +1055,7 @@ namespace Bu | |||
| 822 | */ | 1055 | */ | 
| 823 | const value &operator *() const | 1056 | const value &operator *() const | 
| 824 | { | 1057 | { | 
| 825 | return hsh->getValueAtPos( nPos ); | 1058 | return hsh->core->getValueAtPos( nPos ); | 
| 826 | } | 1059 | } | 
| 827 | 1060 | ||
| 828 | /** | 1061 | /** | 
| @@ -831,7 +1064,7 @@ namespace Bu | |||
| 831 | */ | 1064 | */ | 
| 832 | const key &getKey() const | 1065 | const key &getKey() const | 
| 833 | { | 1066 | { | 
| 834 | return hsh->getKeyAtPos( nPos ); | 1067 | return hsh->core->getKeyAtPos( nPos ); | 
| 835 | } | 1068 | } | 
| 836 | 1069 | ||
| 837 | /** | 1070 | /** | 
| @@ -840,7 +1073,7 @@ namespace Bu | |||
| 840 | */ | 1073 | */ | 
| 841 | const value &getValue() const | 1074 | const value &getValue() const | 
| 842 | { | 1075 | { | 
| 843 | return hsh->getValueAtPos( nPos ); | 1076 | return hsh->core->getValueAtPos( nPos ); | 
| 844 | } | 1077 | } | 
| 845 | } const_iterator; | 1078 | } const_iterator; | 
| 846 | 1079 | ||
| @@ -883,13 +1116,13 @@ namespace Bu | |||
| 883 | { | 1116 | { | 
| 884 | Bu::List<key> lKeys; | 1117 | Bu::List<key> lKeys; | 
| 885 | 1118 | ||
| 886 | for( uint32_t j = 0; j < nCapacity; j++ ) | 1119 | for( uint32_t j = 0; j < core->nCapacity; j++ ) | 
| 887 | { | 1120 | { | 
| 888 | if( isFilled( j ) ) | 1121 | if( core->isFilled( j ) ) | 
| 889 | { | 1122 | { | 
| 890 | if( !isDeleted( j ) ) | 1123 | if( !core->isDeleted( j ) ) | 
| 891 | { | 1124 | { | 
| 892 | lKeys.append( aKeys[j] ); | 1125 | lKeys.append( core->aKeys[j] ); | 
| 893 | } | 1126 | } | 
| 894 | } | 1127 | } | 
| 895 | } | 1128 | } | 
| @@ -901,13 +1134,13 @@ namespace Bu | |||
| 901 | { | 1134 | { | 
| 902 | Bu::List<value> lValues; | 1135 | Bu::List<value> lValues; | 
| 903 | 1136 | ||
| 904 | for( uint32_t j = 0; j < nCapacity; j++ ) | 1137 | for( uint32_t j = 0; j < core->nCapacity; j++ ) | 
| 905 | { | 1138 | { | 
| 906 | if( isFilled( j ) ) | 1139 | if( core->isFilled( j ) ) | 
| 907 | { | 1140 | { | 
| 908 | if( !isDeleted( j ) ) | 1141 | if( !core->isDeleted( j ) ) | 
| 909 | { | 1142 | { | 
| 910 | lValues.append( aValues[j] ); | 1143 | lValues.append( core->aValues[j] ); | 
| 911 | } | 1144 | } | 
| 912 | } | 1145 | } | 
| 913 | } | 1146 | } | 
| @@ -915,244 +1148,74 @@ namespace Bu | |||
| 915 | return lValues; | 1148 | return lValues; | 
| 916 | } | 1149 | } | 
| 917 | 1150 | ||
| 918 | protected: | 1151 | bool operator==( const MyType &rhs ) const | 
| 919 | virtual void onInsert() {} | ||
| 920 | virtual void onUpdate() {} | ||
| 921 | virtual void onDelete() {} | ||
| 922 | virtual void onReHash() {} | ||
| 923 | |||
| 924 | virtual void clearBits() | ||
| 925 | { | ||
| 926 | for( uint32_t j = 0; j < nKeysSize; j++ ) | ||
| 927 | { | ||
| 928 | bFilled[j] = bDeleted[j] = 0; | ||
| 929 | } | ||
| 930 | } | ||
| 931 | |||
| 932 | virtual void fill( uint32_t loc, const key &k, const value &v, uint32_t hash ) | ||
| 933 | { | ||
| 934 | bFilled[loc/32] |= (1<<(loc%32)); | ||
| 935 | va.construct( &aValues[loc], v ); | ||
| 936 | ka.construct( &aKeys[loc], k ); | ||
| 937 | aHashCodes[loc] = hash; | ||
| 938 | nFilled++; | ||
| 939 | //printf("Filled: %d, Deleted: %d, Capacity: %d\n", | ||
| 940 | // nFilled, nDeleted, nCapacity ); | ||
| 941 | } | ||
| 942 | |||
| 943 | virtual void _erase( uint32_t loc ) | ||
| 944 | { | ||
| 945 | bDeleted[loc/32] |= (1<<(loc%32)); | ||
| 946 | va.destroy( &aValues[loc] ); | ||
| 947 | ka.destroy( &aKeys[loc] ); | ||
| 948 | nDeleted++; | ||
| 949 | //printf("Filled: %d, Deleted: %d, Capacity: %d\n", | ||
| 950 | // nFilled, nDeleted, nCapacity ); | ||
| 951 | } | ||
| 952 | |||
| 953 | virtual key &getKeyAtPos( uint32_t nPos ) | ||
| 954 | { | ||
| 955 | return aKeys[nPos]; | ||
| 956 | } | ||
| 957 | |||
| 958 | virtual const key &getKeyAtPos( uint32_t nPos ) const | ||
| 959 | { | ||
| 960 | return aKeys[nPos]; | ||
| 961 | } | ||
| 962 | |||
| 963 | virtual value &getValueAtPos( uint32_t nPos ) | ||
| 964 | { | ||
| 965 | return aValues[nPos]; | ||
| 966 | } | ||
| 967 | |||
| 968 | virtual const value &getValueAtPos( uint32_t nPos ) const | ||
| 969 | { | ||
| 970 | return aValues[nPos]; | ||
| 971 | } | ||
| 972 | |||
| 973 | virtual uint32_t getFirstPos( bool &bFinished ) const | ||
| 974 | { | ||
| 975 | for( uint32_t j = 0; j < nCapacity; j++ ) | ||
| 976 | { | ||
| 977 | if( isFilled( j ) ) | ||
| 978 | if( !isDeleted( j ) ) | ||
| 979 | return j; | ||
| 980 | } | ||
| 981 | |||
| 982 | bFinished = true; | ||
| 983 | return 0; | ||
| 984 | } | ||
| 985 | |||
| 986 | virtual uint32_t getNextPos( uint32_t nPos, bool &bFinished ) const | ||
| 987 | { | 1152 | { | 
| 988 | for( uint32_t j = nPos+1; j < nCapacity; j++ ) | 1153 | if( this == &rhs ) | 
| 989 | { | 1154 | return true; | 
| 990 | if( isFilled( j ) ) | 1155 | if( core == rhs.core ) | 
| 991 | if( !isDeleted( j ) ) | 1156 | return true; | 
| 992 | return j; | 1157 | if( core == NULL || rhs.core == NULL ) | 
| 993 | } | 1158 | return false; | 
| 994 | 1159 | if( getSize() != rhs.getSize() ) | |
| 995 | bFinished = true; | 1160 | return false; | 
| 996 | return 0; | 1161 | |
| 997 | } | 1162 | for( uint32_t j = 0; j < core->nCapacity; j++ ) | 
| 998 | |||
| 999 | uint32_t probe( uint32_t hash, const key &k, bool &bFill, bool rehash=true ) | ||
| 1000 | { | ||
| 1001 | uint32_t nCur = hash%nCapacity; | ||
| 1002 | |||
| 1003 | // First we scan to see if the key is already there, abort if we | ||
| 1004 | // run out of probing room, or we find a non-filled entry | ||
| 1005 | int8_t j; | ||
| 1006 | for( j = 0; | ||
| 1007 | isFilled( nCur ) && j < 32; | ||
| 1008 | nCur = (nCur + (1<<j))%nCapacity, j++ | ||
| 1009 | ) | ||
| 1010 | { | 1163 | { | 
| 1011 | // Is this the same hash code we were looking for? | 1164 | if( core->isFilled( j ) ) | 
| 1012 | if( hash == aHashCodes[nCur] ) | ||
| 1013 | { | 1165 | { | 
| 1014 | // Skip over deleted entries. Deleted entries are also filled, | 1166 | if( !core->isDeleted( j ) ) | 
| 1015 | // so we only have to do this check here. | ||
| 1016 | if( isDeleted( nCur ) ) | ||
| 1017 | continue; | ||
| 1018 | |||
| 1019 | // Is it really the same key? (for safety) | ||
| 1020 | if( __cmpHashKeys( aKeys[nCur], k ) == true ) | ||
| 1021 | { | 1167 | { | 
| 1022 | bFill = true; | 1168 | // Check to see if this key is in the other hash | 
| 1023 | return nCur; | 1169 | if( rhs.has( core->aKeys[j] ) ) | 
| 1170 | { | ||
| 1171 | if( !(core->aValues[j] == rhs.get( core->aKeys[j]) ) ) | ||
| 1172 | { | ||
| 1173 | return false; | ||
| 1174 | } | ||
| 1175 | } | ||
| 1176 | else | ||
| 1177 | { | ||
| 1178 | return false; | ||
| 1179 | } | ||
| 1024 | } | 1180 | } | 
| 1025 | } | 1181 | } | 
| 1026 | } | 1182 | } | 
| 1027 | 1183 | ||
| 1028 | // This is our insurance, if the table is full, then go ahead and | 1184 | return true; | 
| 1029 | // rehash, then try again. | ||
| 1030 | if( (isFilled( nCur ) || j == 32) && rehash == true ) | ||
| 1031 | { | ||
| 1032 | reHash( szCalc(getCapacity(), getFill(), getDeleted()) ); | ||
| 1033 | |||
| 1034 | // This is potentially dangerous, and could cause an infinite loop. | ||
| 1035 | // Be careful writing probe, eh? | ||
| 1036 | return probe( hash, k, bFill ); | ||
| 1037 | } | ||
| 1038 | |||
| 1039 | bFill = false; | ||
| 1040 | return nCur; | ||
| 1041 | } | 1185 | } | 
| 1042 | |||
| 1043 | uint32_t probe( uint32_t hash, key k, bool &bFill ) const | ||
| 1044 | { | ||
| 1045 | uint32_t nCur = hash%nCapacity; | ||
| 1046 | |||
| 1047 | // First we scan to see if the key is already there, abort if we | ||
| 1048 | // run out of probing room, or we find a non-filled entry | ||
| 1049 | for( int8_t j = 0; | ||
| 1050 | isFilled( nCur ) && j < 32; | ||
| 1051 | nCur = (nCur + (1<<j))%nCapacity, j++ | ||
| 1052 | ) | ||
| 1053 | { | ||
| 1054 | // Is this the same hash code we were looking for? | ||
| 1055 | if( hash == aHashCodes[nCur] ) | ||
| 1056 | { | ||
| 1057 | // Skip over deleted entries. Deleted entries are also filled, | ||
| 1058 | // so we only have to do this check here. | ||
| 1059 | if( isDeleted( nCur ) ) | ||
| 1060 | continue; | ||
| 1061 | |||
| 1062 | // Is it really the same key? (for safety) | ||
| 1063 | if( __cmpHashKeys( aKeys[nCur], k ) == true ) | ||
| 1064 | { | ||
| 1065 | bFill = true; | ||
| 1066 | return nCur; | ||
| 1067 | } | ||
| 1068 | } | ||
| 1069 | } | ||
| 1070 | 1186 | ||
| 1071 | bFill = false; | 1187 | bool operator!=( const MyType &rhs ) const | 
| 1072 | return nCur; | 1188 | { | 
| 1189 | return !(*this == rhs); | ||
| 1073 | } | 1190 | } | 
| 1074 | 1191 | ||
| 1075 | void reHash( uint32_t nNewSize ) | 1192 | protected: | 
| 1193 | virtual Core *_copyCore( Core *src ) | ||
| 1076 | { | 1194 | { | 
| 1077 | //printf("---REHASH---"); | 1195 | Core *pRet = _allocateCore(); | 
| 1078 | //printf("Filled: %d, Deleted: %d, Capacity: %d\n", | ||
| 1079 | // nFilled, nDeleted, nCapacity ); | ||
| 1080 | |||
| 1081 | // Save all the old data | ||
| 1082 | uint32_t nOldCapacity = nCapacity; | ||
| 1083 | uint32_t *bOldFilled = bFilled; | ||
| 1084 | uint32_t *aOldHashCodes = aHashCodes; | ||
| 1085 | uint32_t nOldKeysSize = nKeysSize; | ||
| 1086 | uint32_t *bOldDeleted = bDeleted; | ||
| 1087 | value *aOldValues = aValues; | ||
| 1088 | key *aOldKeys = aKeys; | ||
| 1089 | |||
| 1090 | // Calculate new sizes | ||
| 1091 | nCapacity = nNewSize; | ||
| 1092 | nKeysSize = bitsToBytes( nCapacity ); | ||
| 1093 | |||
| 1094 | // Allocate new memory + prep | ||
| 1095 | bFilled = ca.allocate( nKeysSize ); | ||
| 1096 | bDeleted = ca.allocate( nKeysSize ); | ||
| 1097 | clearBits(); | ||
| 1098 | 1196 | ||
| 1099 | aHashCodes = ca.allocate( nCapacity ); | 1197 | pRet->nFilled = 0; | 
| 1100 | aKeys = ka.allocate( nCapacity ); | 1198 | pRet->nDeleted = 0; | 
| 1101 | aValues = va.allocate( nCapacity ); | 1199 | pRet->nCapacity = src->nCapacity; | 
| 1102 | 1200 | pRet->nKeysSize = bitsToBytes( pRet->nCapacity ); | |
| 1103 | nDeleted = nFilled = 0; | 1201 | pRet->bFilled = pRet->ca.allocate( pRet->nKeysSize ); | 
| 1202 | pRet->bDeleted = pRet->ca.allocate( pRet->nKeysSize ); | ||
| 1203 | pRet->clearBits(); | ||
| 1104 | 1204 | ||
| 1105 | // Re-insert all of the old data (except deleted items) | 1205 | pRet->aHashCodes = pRet->ca.allocate( pRet->nCapacity ); | 
| 1106 | for( uint32_t j = 0; j < nOldCapacity; j++ ) | 1206 | pRet->aKeys = pRet->ka.allocate( pRet->nCapacity ); | 
| 1107 | { | 1207 | pRet->aValues = pRet->va.allocate( pRet->nCapacity ); | 
| 1108 | if( (bOldFilled[j/32]&(1<<(j%32)))!=0 && | ||
| 1109 | (bOldDeleted[j/32]&(1<<(j%32)))==0 ) | ||
| 1110 | { | ||
| 1111 | insert( aOldKeys[j], aOldValues[j] ); | ||
| 1112 | } | ||
| 1113 | } | ||
| 1114 | 1208 | ||
| 1115 | // Delete all of the old data | 1209 | for( uint32_t j = 0; j < src->nCapacity; j++ ) | 
| 1116 | for( uint32_t j = 0; j < nOldCapacity; j++ ) | ||
| 1117 | { | 1210 | { | 
| 1118 | if( (bOldFilled[j/32]&(1<<(j%32)))!=0 && | 1211 | if( src->isFilled( j ) && !src->isDeleted( j ) ) | 
| 1119 | (bOldDeleted[j/32]&(1<<(j%32)))==0 ) | ||
| 1120 | { | 1212 | { | 
| 1121 | va.destroy( &aOldValues[j] ); | 1213 | pRet->insert( src->aKeys[j], src->aValues[j] ); | 
| 1122 | ka.destroy( &aOldKeys[j] ); | ||
| 1123 | } | 1214 | } | 
| 1124 | } | 1215 | } | 
| 1125 | va.deallocate( aOldValues, nOldCapacity ); | ||
| 1126 | ka.deallocate( aOldKeys, nOldCapacity ); | ||
| 1127 | ca.deallocate( bOldFilled, nOldKeysSize ); | ||
| 1128 | ca.deallocate( bOldDeleted, nOldKeysSize ); | ||
| 1129 | ca.deallocate( aOldHashCodes, nOldCapacity ); | ||
| 1130 | } | ||
| 1131 | 1216 | ||
| 1132 | virtual bool isFilled( uint32_t loc ) const | 1217 | return pRet; | 
| 1133 | { | ||
| 1134 | return (bFilled[loc/32]&(1<<(loc%32)))!=0; | ||
| 1135 | } | ||
| 1136 | |||
| 1137 | virtual bool isDeleted( uint32_t loc ) const | ||
| 1138 | { | ||
| 1139 | return (bDeleted[loc/32]&(1<<(loc%32)))!=0; | ||
| 1140 | } | 1218 | } | 
| 1141 | |||
| 1142 | protected: | ||
| 1143 | uint32_t nCapacity; | ||
| 1144 | uint32_t nFilled; | ||
| 1145 | uint32_t nDeleted; | ||
| 1146 | uint32_t *bFilled; | ||
| 1147 | uint32_t *bDeleted; | ||
| 1148 | uint32_t nKeysSize; | ||
| 1149 | key *aKeys; | ||
| 1150 | value *aValues; | ||
| 1151 | uint32_t *aHashCodes; | ||
| 1152 | valuealloc va; | ||
| 1153 | keyalloc ka; | ||
| 1154 | challoc ca; | ||
| 1155 | sizecalc szCalc; | ||
| 1156 | }; | 1219 | }; | 
| 1157 | 1220 | ||
| 1158 | template<typename T> uint32_t __calcHashCode( const T &k ) | 1221 | template<typename T> uint32_t __calcHashCode( const T &k ) | 
| @@ -13,76 +13,93 @@ | |||
| 13 | #include "bu/exceptionbase.h" | 13 | #include "bu/exceptionbase.h" | 
| 14 | #include "bu/util.h" | 14 | #include "bu/util.h" | 
| 15 | #include "bu/queue.h" | 15 | #include "bu/queue.h" | 
| 16 | #include "bu/sharedcore.h" | ||
| 16 | 17 | ||
| 17 | namespace Bu | 18 | namespace Bu | 
| 18 | { | 19 | { | 
| 19 | subExceptionDecl( HeapException ); | 20 | subExceptionDecl( HeapException ); | 
| 20 | 21 | ||
| 21 | /** | 22 | template<typename item, typename cmpfunc, typename itemalloc> | 
| 22 | * A priority queue that allows for an unlimited number of priorities. All | 23 | class Heap; | 
| 23 | * objects enqueued must support less-than-comparison. Then every time an | 24 | |
| 24 | * item is dequeued it is always the least item in the heap. The heap | 25 | /** @cond DEVEL */ | 
| 25 | * operates using a binary tree for storage, which allows most operations | 26 | template<typename item, typename cmpfunc, typename itemalloc> | 
| 26 | * to be very fast. Enqueueing and dequeueing are both O(log(N)) operatoins | 27 | class HeapCore | 
| 27 | * whereas peeking is constant time. | ||
| 28 | * | ||
| 29 | * This heap implementation allows iterating, however please note that any | ||
| 30 | * enqueue or dequeue operation will invalidate the iterator and make it | ||
| 31 | * unusable (if it still works, you shouldn't trust the results). Also, | ||
| 32 | * the items are not stored in memory in order, they are optomized into a | ||
| 33 | * tree. This means that the items will be in effectively random order | ||
| 34 | * while iterating through them, and the order cannot be trusted. Also, | ||
| 35 | * modifying an item in the heap will not cause that item to be re-sorted. | ||
| 36 | * If you want to change the position of an item in the heap you will have | ||
| 37 | * to dequeue every item before it, dequeue that item, change it, and | ||
| 38 | * re-enqueue all of the items removed. | ||
| 39 | */ | ||
| 40 | template<typename item, typename cmpfunc=__basicLTCmp<item>, typename itemalloc=std::allocator<item> > | ||
| 41 | class Heap : public Queue<item> | ||
| 42 | { | 28 | { | 
| 43 | public: | 29 | friend class Heap<item, cmpfunc, itemalloc>; | 
| 44 | Heap() : | 30 | friend class SharedCore< | 
| 45 | iSize( 7 ), | 31 | Heap<item, cmpfunc, itemalloc>, HeapCore<item, cmpfunc, itemalloc> | 
| 32 | >; | ||
| 33 | private: | ||
| 34 | HeapCore() : | ||
| 35 | iSize( 0 ), | ||
| 46 | iFill( 0 ), | 36 | iFill( 0 ), | 
| 47 | aItem( ia.allocate( iSize ) ) | 37 | aItem( NULL ) | 
| 48 | { | 38 | { | 
| 49 | } | 39 | } | 
| 50 | 40 | ||
| 51 | Heap( cmpfunc cmpin ) : | 41 | virtual ~HeapCore() | 
| 52 | iSize( 7 ), | ||
| 53 | iFill( 0 ), | ||
| 54 | aItem( ia.allocate( iSize ) ), | ||
| 55 | cmp( cmpin ) | ||
| 56 | { | 42 | { | 
| 43 | clear(); | ||
| 57 | } | 44 | } | 
| 58 | 45 | ||
| 59 | Heap( int iInitialCapacity ) : | 46 | void init() | 
| 60 | iSize( 0 ), | ||
| 61 | iFill( 0 ), | ||
| 62 | aItem( NULL ) | ||
| 63 | { | 47 | { | 
| 64 | for( iSize = 1; iSize < iInitialCapacity; iSize=iSize*2+1 ) { } | 48 | if( iSize > 0 ) | 
| 49 | return; | ||
| 50 | |||
| 51 | iSize = 7; | ||
| 52 | iFill = 0; | ||
| 65 | aItem = ia.allocate( iSize ); | 53 | aItem = ia.allocate( iSize ); | 
| 66 | } | 54 | } | 
| 67 | 55 | ||
| 68 | Heap( cmpfunc cmpin, int iInitialCapacity ) : | 56 | void init( int iCap ) | 
| 69 | iSize( 0 ), | ||
| 70 | iFill( 0 ), | ||
| 71 | aItem( NULL ), | ||
| 72 | cmp( cmpin ) | ||
| 73 | { | 57 | { | 
| 74 | for( iSize = 1; iSize < iInitialCapacity; iSize=iSize*2+1 ) { } | 58 | if( iSize > 0 ) | 
| 59 | return; | ||
| 60 | |||
| 61 | for( iSize = 1; iSize < iCap; iSize=iSize*2+1 ) { } | ||
| 62 | iFill = 0; | ||
| 75 | aItem = ia.allocate( iSize ); | 63 | aItem = ia.allocate( iSize ); | 
| 76 | } | 64 | } | 
| 77 | 65 | ||
| 78 | virtual ~Heap() | 66 | void clear() | 
| 79 | { | 67 | { | 
| 68 | if( iSize == 0 ) | ||
| 69 | return; | ||
| 70 | |||
| 80 | for( int j = 0; j < iFill; j++ ) | 71 | for( int j = 0; j < iFill; j++ ) | 
| 81 | ia.destroy( &aItem[j] ); | 72 | ia.destroy( &aItem[j] ); | 
| 82 | ia.deallocate( aItem, iSize ); | 73 | ia.deallocate( aItem, iSize ); | 
| 83 | aItem = NULL; | 74 | aItem = NULL; | 
| 75 | iSize = 0; | ||
| 76 | iFill = 0; | ||
| 84 | } | 77 | } | 
| 78 | |||
| 79 | void upSize() | ||
| 80 | { | ||
| 81 | if( iSize == 0 ) | ||
| 82 | { | ||
| 83 | init(); | ||
| 84 | return; | ||
| 85 | } | ||
| 85 | 86 | ||
| 87 | item *aNewItems = ia.allocate( iSize*2+1 ); | ||
| 88 | // | ||
| 89 | // We cannot use a memcopy here because we don't know what kind | ||
| 90 | // of datastructures are being used, we have to copy them one at | ||
| 91 | // a time. | ||
| 92 | // | ||
| 93 | for( int j = 0; j < iFill; j++ ) | ||
| 94 | { | ||
| 95 | ia.construct( &aNewItems[j], aItem[j] ); | ||
| 96 | ia.destroy( &aItem[j] ); | ||
| 97 | } | ||
| 98 | ia.deallocate( aItem, iSize ); | ||
| 99 | aItem = aNewItems; | ||
| 100 | iSize = iSize*2+1; | ||
| 101 | } | ||
| 102 | |||
| 86 | virtual void enqueue( const item &it ) | 103 | virtual void enqueue( const item &it ) | 
| 87 | { | 104 | { | 
| 88 | item i = it; // TODO: This is a silly workaround, put the i item | 105 | item i = it; // TODO: This is a silly workaround, put the i item | 
| @@ -126,20 +143,6 @@ namespace Bu | |||
| 126 | iFill++; | 143 | iFill++; | 
| 127 | } | 144 | } | 
| 128 | 145 | ||
| 129 | virtual item &peek() | ||
| 130 | { | ||
| 131 | if( iFill == 0 ) | ||
| 132 | throw HeapException("Heap empty."); | ||
| 133 | return aItem[0]; | ||
| 134 | } | ||
| 135 | |||
| 136 | virtual const item &peek() const | ||
| 137 | { | ||
| 138 | if( iFill == 0 ) | ||
| 139 | throw HeapException("Heap empty."); | ||
| 140 | return aItem[0]; | ||
| 141 | } | ||
| 142 | |||
| 143 | virtual item dequeue() | 146 | virtual item dequeue() | 
| 144 | { | 147 | { | 
| 145 | if( iFill == 0 ) | 148 | if( iFill == 0 ) | 
| @@ -174,14 +177,118 @@ namespace Bu | |||
| 174 | return iRet; | 177 | return iRet; | 
| 175 | } | 178 | } | 
| 176 | 179 | ||
| 180 | private: | ||
| 181 | int iSize; | ||
| 182 | int iFill; | ||
| 183 | item *aItem; | ||
| 184 | cmpfunc cmp; | ||
| 185 | itemalloc ia; | ||
| 186 | }; | ||
| 187 | /** @endcond */ | ||
| 188 | |||
| 189 | /** | ||
| 190 | * A priority queue that allows for an unlimited number of priorities. All | ||
| 191 | * objects enqueued must support less-than-comparison. Then every time an | ||
| 192 | * item is dequeued it is always the least item in the heap. The heap | ||
| 193 | * operates using a binary tree for storage, which allows most operations | ||
| 194 | * to be very fast. Enqueueing and dequeueing are both O(log(N)) operatoins | ||
| 195 | * whereas peeking is constant time. | ||
| 196 | * | ||
| 197 | * This heap implementation allows iterating, however please note that any | ||
| 198 | * enqueue or dequeue operation will invalidate the iterator and make it | ||
| 199 | * unusable (if it still works, you shouldn't trust the results). Also, | ||
| 200 | * the items are not stored in memory in order, they are optomized into a | ||
| 201 | * tree. This means that the items will be in effectively random order | ||
| 202 | * while iterating through them, and the order cannot be trusted. Also, | ||
| 203 | * modifying an item in the heap will not cause that item to be re-sorted. | ||
| 204 | * If you want to change the position of an item in the heap you will have | ||
| 205 | * to dequeue every item before it, dequeue that item, change it, and | ||
| 206 | * re-enqueue all of the items removed. | ||
| 207 | */ | ||
| 208 | template<typename item, typename cmpfunc=__basicLTCmp<item>, typename itemalloc=std::allocator<item> > | ||
| 209 | class Heap : public Queue<item>, public SharedCore< | ||
| 210 | Heap<item, cmpfunc, itemalloc>, | ||
| 211 | HeapCore<item, cmpfunc, itemalloc> | ||
| 212 | > | ||
| 213 | { | ||
| 214 | private: | ||
| 215 | typedef class Heap<item,cmpfunc,itemalloc> MyType; | ||
| 216 | typedef class HeapCore<item,cmpfunc,itemalloc> Core; | ||
| 217 | |||
| 218 | protected: | ||
| 219 | using SharedCore<MyType, Core>::core; | ||
| 220 | using SharedCore<MyType, Core>::_hardCopy; | ||
| 221 | using SharedCore<MyType, Core>::_resetCore; | ||
| 222 | using SharedCore<MyType, Core>::_allocateCore; | ||
| 223 | |||
| 224 | public: | ||
| 225 | Heap() | ||
| 226 | { | ||
| 227 | } | ||
| 228 | |||
| 229 | Heap( cmpfunc cmpin ) | ||
| 230 | { | ||
| 231 | core->cmp = cmpin; | ||
| 232 | } | ||
| 233 | |||
| 234 | Heap( int iInitialCapacity ) | ||
| 235 | { | ||
| 236 | core->init( iInitialCapacity ); | ||
| 237 | } | ||
| 238 | |||
| 239 | Heap( cmpfunc cmpin, int iInitialCapacity ) | ||
| 240 | { | ||
| 241 | core->cmp = cmpin; | ||
| 242 | core->init( iInitialCapacity ); | ||
| 243 | } | ||
| 244 | |||
| 245 | Heap( const MyType &rSrc ) : | ||
| 246 | SharedCore<MyType, Core>( rSrc ) | ||
| 247 | { | ||
| 248 | } | ||
| 249 | |||
| 250 | virtual ~Heap() | ||
| 251 | { | ||
| 252 | } | ||
| 253 | |||
| 254 | virtual void enqueue( const item &it ) | ||
| 255 | { | ||
| 256 | _hardCopy(); | ||
| 257 | |||
| 258 | core->enqueue( it ); | ||
| 259 | } | ||
| 260 | |||
| 261 | virtual item &peek() | ||
| 262 | { | ||
| 263 | _hardCopy(); | ||
| 264 | |||
| 265 | if( core->iFill == 0 ) | ||
| 266 | throw HeapException("Heap empty."); | ||
| 267 | return core->aItem[0]; | ||
| 268 | } | ||
| 269 | |||
| 270 | virtual const item &peek() const | ||
| 271 | { | ||
| 272 | if( core->iFill == 0 ) | ||
| 273 | throw HeapException("Heap empty."); | ||
| 274 | return core->aItem[0]; | ||
| 275 | } | ||
| 276 | |||
| 277 | virtual item dequeue() | ||
| 278 | { | ||
| 279 | _hardCopy(); | ||
| 280 | |||
| 281 | return core->dequeue(); | ||
| 282 | } | ||
| 283 | |||
| 177 | virtual bool isEmpty() const | 284 | virtual bool isEmpty() const | 
| 178 | { | 285 | { | 
| 179 | return (iFill==0); | 286 | return (core->iFill==0); | 
| 180 | } | 287 | } | 
| 181 | 288 | ||
| 182 | virtual int getSize() const | 289 | virtual int getSize() const | 
| 183 | { | 290 | { | 
| 184 | return iFill; | 291 | return core->iFill; | 
| 185 | } | 292 | } | 
| 186 | 293 | ||
| 187 | class iterator | 294 | class iterator | 
| @@ -201,7 +308,7 @@ namespace Bu | |||
| 201 | { | 308 | { | 
| 202 | if( pHeap == NULL ) | 309 | if( pHeap == NULL ) | 
| 203 | throw Bu::ExceptionBase("Iterator not initialized."); | 310 | throw Bu::ExceptionBase("Iterator not initialized."); | 
| 204 | if( iIndex < 0 || iIndex >= pHeap->iFill ) | 311 | if( iIndex < 0 || iIndex >= pHeap->core->iFill ) | 
| 205 | throw Bu::ExceptionBase("Iterator out of bounds."); | 312 | throw Bu::ExceptionBase("Iterator out of bounds."); | 
| 206 | } | 313 | } | 
| 207 | 314 | ||
| @@ -230,12 +337,16 @@ namespace Bu | |||
| 230 | 337 | ||
| 231 | item &operator*() | 338 | item &operator*() | 
| 232 | { | 339 | { | 
| 233 | return pHeap->aItem[iIndex]; | 340 | pHeap->_hardCopy(); | 
| 341 | |||
| 342 | return pHeap->core->aItem[iIndex]; | ||
| 234 | } | 343 | } | 
| 235 | 344 | ||
| 236 | item *operator->() | 345 | item *operator->() | 
| 237 | { | 346 | { | 
| 238 | return &(pHeap->aItem[iIndex]); | 347 | pHeap->_hardCopy(); | 
| 348 | |||
| 349 | return &(pHeap->core->aItem[iIndex]); | ||
| 239 | } | 350 | } | 
| 240 | 351 | ||
| 241 | iterator &operator++() | 352 | iterator &operator++() | 
| @@ -260,7 +371,7 @@ namespace Bu | |||
| 260 | { | 371 | { | 
| 261 | checkValid(); | 372 | checkValid(); | 
| 262 | iIndex++; | 373 | iIndex++; | 
| 263 | if( iIndex >= pHeap->iFill ) | 374 | if( iIndex >= pHeap->core->iFill ) | 
| 264 | iIndex = -1; | 375 | iIndex = -1; | 
| 265 | 376 | ||
| 266 | return *this; | 377 | return *this; | 
| @@ -279,7 +390,7 @@ namespace Bu | |||
| 279 | checkValid(); | 390 | checkValid(); | 
| 280 | iterator ret( *this ); | 391 | iterator ret( *this ); | 
| 281 | ret.iIndex += iDelta; | 392 | ret.iIndex += iDelta; | 
| 282 | if( ret.iIndex >= pHeap->iFill ) | 393 | if( ret.iIndex >= pHeap->core->iFill ) | 
| 283 | ret.iIndex = -1; | 394 | ret.iIndex = -1; | 
| 284 | return ret; | 395 | return ret; | 
| 285 | } | 396 | } | 
| @@ -294,12 +405,12 @@ namespace Bu | |||
| 294 | return ret; | 405 | return ret; | 
| 295 | } | 406 | } | 
| 296 | 407 | ||
| 297 | operator bool() | 408 | operator bool() const | 
| 298 | { | 409 | { | 
| 299 | return iIndex != -1; | 410 | return iIndex != -1; | 
| 300 | } | 411 | } | 
| 301 | 412 | ||
| 302 | bool isValid() | 413 | bool isValid() const | 
| 303 | { | 414 | { | 
| 304 | return iIndex != -1; | 415 | return iIndex != -1; | 
| 305 | } | 416 | } | 
| @@ -328,7 +439,7 @@ namespace Bu | |||
| 328 | { | 439 | { | 
| 329 | if( pHeap == NULL ) | 440 | if( pHeap == NULL ) | 
| 330 | throw Bu::ExceptionBase("Iterator not initialized."); | 441 | throw Bu::ExceptionBase("Iterator not initialized."); | 
| 331 | if( iIndex < 0 || iIndex >= pHeap->iFill ) | 442 | if( iIndex < 0 || iIndex >= pHeap->core->iFill ) | 
| 332 | throw Bu::ExceptionBase("Iterator out of bounds."); | 443 | throw Bu::ExceptionBase("Iterator out of bounds."); | 
| 333 | } | 444 | } | 
| 334 | 445 | ||
| @@ -363,19 +474,23 @@ namespace Bu | |||
| 363 | 474 | ||
| 364 | const item &operator*() | 475 | const item &operator*() | 
| 365 | { | 476 | { | 
| 366 | return pHeap->aItem[iIndex]; | 477 | pHeap->_hardCopy(); | 
| 478 | |||
| 479 | return pHeap->core->aItem[iIndex]; | ||
| 367 | } | 480 | } | 
| 368 | 481 | ||
| 369 | const item *operator->() | 482 | const item *operator->() | 
| 370 | { | 483 | { | 
| 371 | return &(pHeap->aItem[iIndex]); | 484 | pHeap->_hardCopy(); | 
| 485 | |||
| 486 | return &(pHeap->core->aItem[iIndex]); | ||
| 372 | } | 487 | } | 
| 373 | 488 | ||
| 374 | const_iterator &operator++() | 489 | const_iterator &operator++() | 
| 375 | { | 490 | { | 
| 376 | checkValid(); | 491 | checkValid(); | 
| 377 | iIndex++; | 492 | iIndex++; | 
| 378 | if( iIndex >= pHeap->iFill ) | 493 | if( iIndex >= pHeap->core->iFill ) | 
| 379 | iIndex = -1; | 494 | iIndex = -1; | 
| 380 | 495 | ||
| 381 | return *this; | 496 | return *this; | 
| @@ -393,7 +508,7 @@ namespace Bu | |||
| 393 | { | 508 | { | 
| 394 | checkValid(); | 509 | checkValid(); | 
| 395 | iIndex++; | 510 | iIndex++; | 
| 396 | if( iIndex >= pHeap->iFill ) | 511 | if( iIndex >= pHeap->core->iFill ) | 
| 397 | iIndex = -1; | 512 | iIndex = -1; | 
| 398 | 513 | ||
| 399 | return *this; | 514 | return *this; | 
| @@ -427,12 +542,12 @@ namespace Bu | |||
| 427 | return ret; | 542 | return ret; | 
| 428 | } | 543 | } | 
| 429 | 544 | ||
| 430 | operator bool() | 545 | operator bool() const | 
| 431 | { | 546 | { | 
| 432 | return iIndex != -1; | 547 | return iIndex != -1; | 
| 433 | } | 548 | } | 
| 434 | 549 | ||
| 435 | bool isValid() | 550 | bool isValid() const | 
| 436 | { | 551 | { | 
| 437 | return iIndex != -1; | 552 | return iIndex != -1; | 
| 438 | } | 553 | } | 
| @@ -452,14 +567,14 @@ namespace Bu | |||
| 452 | 567 | ||
| 453 | iterator begin() | 568 | iterator begin() | 
| 454 | { | 569 | { | 
| 455 | if( iFill == 0 ) | 570 | if( core->iFill == 0 ) | 
| 456 | return end(); | 571 | return end(); | 
| 457 | return iterator( this, 0 ); | 572 | return iterator( this, 0 ); | 
| 458 | } | 573 | } | 
| 459 | 574 | ||
| 460 | const_iterator begin() const | 575 | const_iterator begin() const | 
| 461 | { | 576 | { | 
| 462 | if( iFill == 0 ) | 577 | if( core->iFill == 0 ) | 
| 463 | return end(); | 578 | return end(); | 
| 464 | return const_iterator( this, 0 ); | 579 | return const_iterator( this, 0 ); | 
| 465 | } | 580 | } | 
| @@ -475,31 +590,22 @@ namespace Bu | |||
| 475 | } | 590 | } | 
| 476 | 591 | ||
| 477 | 592 | ||
| 478 | private: | 593 | protected: | 
| 479 | void upSize() | 594 | virtual Core *_copyCore( Core *src ) | 
| 480 | { | 595 | { | 
| 481 | item *aNewItems = ia.allocate( iSize*2+1 ); | 596 | Core *pRet = _allocateCore(); | 
| 482 | // | 597 | |
| 483 | // We cannot use a memcopy here because we don't know what kind | 598 | pRet->iSize = src->iSize; | 
| 484 | // of datastructures are being used, we have to copy them one at | 599 | pRet->iFill = src->iFill; | 
| 485 | // a time. | 600 | pRet->cmp = src->cmp; | 
| 486 | // | 601 | pRet->aItem = pRet->ia.allocate( pRet->iSize ); | 
| 487 | for( int j = 0; j < iFill; j++ ) | 602 | for( int j = 0; j < pRet->iFill; j++ ) | 
| 488 | { | 603 | { | 
| 489 | ia.construct( &aNewItems[j], aItem[j] ); | 604 | pRet->ia.construct( &pRet->aItem[j], src->aItem[j] ); | 
| 490 | ia.destroy( &aItem[j] ); | ||
| 491 | } | 605 | } | 
| 492 | ia.deallocate( aItem, iSize ); | ||
| 493 | aItem = aNewItems; | ||
| 494 | iSize = iSize*2+1; | ||
| 495 | } | ||
| 496 | 606 | ||
| 497 | private: | 607 | return pRet; | 
| 498 | int iSize; | 608 | } | 
| 499 | int iFill; | ||
| 500 | item *aItem; | ||
| 501 | cmpfunc cmp; | ||
| 502 | itemalloc ia; | ||
| 503 | }; | 609 | }; | 
| 504 | }; | 610 | }; | 
| 505 | 611 | ||
| diff --git a/src/httpget.h b/src/httpget.h index 410914c..783f880 100644 --- a/src/httpget.h +++ b/src/httpget.h | |||
| @@ -11,7 +11,7 @@ | |||
| 11 | #include "bu/stream.h" | 11 | #include "bu/stream.h" | 
| 12 | #include "bu/fstring.h" | 12 | #include "bu/fstring.h" | 
| 13 | #include "bu/url.h" | 13 | #include "bu/url.h" | 
| 14 | #include "bu/socket.h" | 14 | #include "bu/tcpsocket.h" | 
| 15 | #include "bu/hash.h" | 15 | #include "bu/hash.h" | 
| 16 | 16 | ||
| 17 | namespace Bu | 17 | namespace Bu | 
| @@ -52,7 +52,7 @@ namespace Bu | |||
| 52 | private: | 52 | private: | 
| 53 | Bu::Url uSrc; | 53 | Bu::Url uSrc; | 
| 54 | Bu::FString sMethod; | 54 | Bu::FString sMethod; | 
| 55 | Bu::Socket sSrv; | 55 | Bu::TcpSocket sSrv; | 
| 56 | typedef Bu::Hash<Bu::FString,Bu::FString> MimeHash; | 56 | typedef Bu::Hash<Bu::FString,Bu::FString> MimeHash; | 
| 57 | MimeHash hMimeIn; | 57 | MimeHash hMimeIn; | 
| 58 | MimeHash hMimeOut; | 58 | MimeHash hMimeOut; | 
| diff --git a/src/itoserver.cpp b/src/itoserver.cpp index a1d804a..ea737bf 100644 --- a/src/itoserver.cpp +++ b/src/itoserver.cpp | |||
| @@ -7,9 +7,9 @@ | |||
| 7 | 7 | ||
| 8 | #include "bu/itoserver.h" | 8 | #include "bu/itoserver.h" | 
| 9 | #include <errno.h> | 9 | #include <errno.h> | 
| 10 | #include "bu/serversocket.h" | 10 | #include "bu/tcpserversocket.h" | 
| 11 | #include "bu/client.h" | 11 | #include "bu/client.h" | 
| 12 | #include "bu/socket.h" | 12 | #include "bu/tcpsocket.h" | 
| 13 | 13 | ||
| 14 | #include "bu/config.h" | 14 | #include "bu/config.h" | 
| 15 | 15 | ||
| @@ -41,7 +41,7 @@ Bu::ItoServer::~ItoServer() | |||
| 41 | 41 | ||
| 42 | void Bu::ItoServer::addPort( int nPort, int nPoolSize ) | 42 | void Bu::ItoServer::addPort( int nPort, int nPoolSize ) | 
| 43 | { | 43 | { | 
| 44 | ServerSocket *s = new ServerSocket( nPort, nPoolSize ); | 44 | TcpServerSocket *s = new TcpServerSocket( nPort, nPoolSize ); | 
| 45 | int nSocket = s->getSocket(); | 45 | int nSocket = s->getSocket(); | 
| 46 | FD_SET( nSocket, &fdActive ); | 46 | FD_SET( nSocket, &fdActive ); | 
| 47 | hServers.insert( nSocket, s ); | 47 | hServers.insert( nSocket, s ); | 
| @@ -49,7 +49,7 @@ void Bu::ItoServer::addPort( int nPort, int nPoolSize ) | |||
| 49 | 49 | ||
| 50 | void Bu::ItoServer::addPort( const FString &sAddr, int nPort, int nPoolSize ) | 50 | void Bu::ItoServer::addPort( const FString &sAddr, int nPort, int nPoolSize ) | 
| 51 | { | 51 | { | 
| 52 | ServerSocket *s = new ServerSocket( sAddr, nPort, nPoolSize ); | 52 | TcpServerSocket *s = new TcpServerSocket( sAddr, nPort, nPoolSize ); | 
| 53 | int nSocket = s->getSocket(); | 53 | int nSocket = s->getSocket(); | 
| 54 | FD_SET( nSocket, &fdActive ); | 54 | FD_SET( nSocket, &fdActive ); | 
| 55 | hServers.insert( nSocket, s ); | 55 | hServers.insert( nSocket, s ); | 
| @@ -92,7 +92,7 @@ void Bu::ItoServer::run() | |||
| 92 | { | 92 | { | 
| 93 | if( FD_ISSET( i.getKey(), &fdRead ) ) | 93 | if( FD_ISSET( i.getKey(), &fdRead ) ) | 
| 94 | { | 94 | { | 
| 95 | ServerSocket *pSrv = i.getValue(); | 95 | TcpServerSocket *pSrv = i.getValue(); | 
| 96 | addClient( pSrv->accept(), pSrv->getPort() ); | 96 | addClient( pSrv->accept(), pSrv->getPort() ); | 
| 97 | } | 97 | } | 
| 98 | } | 98 | } | 
| @@ -126,7 +126,7 @@ Bu::ItoServer::ItoClient::ItoClient( ItoServer &rSrv, int iSocket, int iPort, | |||
| 126 | FD_SET( iSocket, &fdActive ); | 126 | FD_SET( iSocket, &fdActive ); | 
| 127 | 127 | ||
| 128 | pClient = new Client( | 128 | pClient = new Client( | 
| 129 | new Bu::Socket( iSocket ), | 129 | new Bu::TcpSocket( iSocket ), | 
| 130 | new SrvClientLinkFactory( rSrv ) | 130 | new SrvClientLinkFactory( rSrv ) | 
| 131 | ); | 131 | ); | 
| 132 | } | 132 | } | 
| diff --git a/src/itoserver.h b/src/itoserver.h index c08d453..81e42cc 100644 --- a/src/itoserver.h +++ b/src/itoserver.h | |||
| @@ -26,8 +26,8 @@ | |||
| 26 | 26 | ||
| 27 | namespace Bu | 27 | namespace Bu | 
| 28 | { | 28 | { | 
| 29 | class ServerSocket; | 29 | class TcpServerSocket; | 
| 30 | class Socket; | 30 | class TcpSocket; | 
| 31 | class Client; | 31 | class Client; | 
| 32 | 32 | ||
| 33 | /** | 33 | /** | 
| @@ -126,7 +126,7 @@ namespace Bu | |||
| 126 | int nTimeoutSec; | 126 | int nTimeoutSec; | 
| 127 | int nTimeoutUSec; | 127 | int nTimeoutUSec; | 
| 128 | fd_set fdActive; | 128 | fd_set fdActive; | 
| 129 | typedef Hash<int,ServerSocket *> ServerHash; | 129 | typedef Hash<int,TcpServerSocket *> ServerHash; | 
| 130 | ServerHash hServers; | 130 | ServerHash hServers; | 
| 131 | typedef Hash<int,ItoClient *> ClientHash; | 131 | typedef Hash<int,ItoClient *> ClientHash; | 
| 132 | typedef ItoQueue<ItoClient *> ClientQueue; | 132 | typedef ItoQueue<ItoClient *> ClientQueue; | 
| diff --git a/src/lexer.h b/src/lexer.h index 2cb5f51..9840afe 100644 --- a/src/lexer.h +++ b/src/lexer.h | |||
| @@ -36,6 +36,10 @@ namespace Bu | |||
| 36 | } | 36 | } | 
| 37 | TokenType iToken; | 37 | TokenType iToken; | 
| 38 | Bu::Variant vExtra; | 38 | Bu::Variant vExtra; | 
| 39 | int iStartCol; | ||
| 40 | int iStartRow; | ||
| 41 | int iEndCol; | ||
| 42 | int iEndRow; | ||
| 39 | }; | 43 | }; | 
| 40 | 44 | ||
| 41 | virtual Token *nextToken()=0; | 45 | virtual Token *nextToken()=0; | 
| @@ -16,6 +16,7 @@ | |||
| 16 | 16 | ||
| 17 | namespace Bu | 17 | namespace Bu | 
| 18 | { | 18 | { | 
| 19 | /** @cond DEVEL */ | ||
| 19 | template<typename value> | 20 | template<typename value> | 
| 20 | struct ListLink | 21 | struct ListLink | 
| 21 | { | 22 | { | 
| @@ -24,10 +25,18 @@ namespace Bu | |||
| 24 | ListLink *pPrev; | 25 | ListLink *pPrev; | 
| 25 | }; | 26 | }; | 
| 26 | 27 | ||
| 27 | template<typename value, typename valuealloc, | 28 | template<typename value, typename valuealloc, typename linkalloc> | 
| 28 | typename linkalloc> | 29 | class List; | 
| 30 | |||
| 31 | template<typename value, typename valuealloc, typename linkalloc> | ||
| 29 | struct ListCore | 32 | struct ListCore | 
| 30 | { | 33 | { | 
| 34 | friend class List<value, valuealloc, linkalloc>; | ||
| 35 | friend class SharedCore< | ||
| 36 | List<value, valuealloc, linkalloc>, | ||
| 37 | ListCore<value, valuealloc, linkalloc> | ||
| 38 | >; | ||
| 39 | private: | ||
| 31 | typedef struct ListLink<value> Link; | 40 | typedef struct ListLink<value> Link; | 
| 32 | ListCore() : | 41 | ListCore() : | 
| 33 | pFirst( NULL ), | 42 | pFirst( NULL ), | 
| @@ -177,6 +186,7 @@ namespace Bu | |||
| 177 | } | 186 | } | 
| 178 | } | 187 | } | 
| 179 | }; | 188 | }; | 
| 189 | /** @endcond */ | ||
| 180 | 190 | ||
| 181 | /** | 191 | /** | 
| 182 | * Linked list template container. This class is similar to the stl list | 192 | * Linked list template container. This class is similar to the stl list | 
| @@ -189,12 +199,15 @@ namespace Bu | |||
| 189 | *@param value (typename) The type of data to store in your list | 199 | *@param value (typename) The type of data to store in your list | 
| 190 | *@param valuealloc (typename) Memory Allocator for your value type | 200 | *@param valuealloc (typename) Memory Allocator for your value type | 
| 191 | *@param linkalloc (typename) Memory Allocator for the list links. | 201 | *@param linkalloc (typename) Memory Allocator for the list links. | 
| 202 | *@extends SharedCore | ||
| 192 | *@ingroup Containers | 203 | *@ingroup Containers | 
| 193 | */ | 204 | */ | 
| 194 | template<typename value, typename valuealloc=std::allocator<value>, | 205 | template<typename value, typename valuealloc=std::allocator<value>, | 
| 195 | typename linkalloc=std::allocator<struct ListLink<value> > > | 206 | typename linkalloc=std::allocator<struct ListLink<value> > > | 
| 196 | class List : public SharedCore< struct ListCore<value, valuealloc, | 207 | class List /** @cond */ : public SharedCore< | 
| 197 | linkalloc> > | 208 | List<value, valuealloc, linkalloc>, | 
| 209 | ListCore<value, valuealloc, linkalloc> | ||
| 210 | > /** @endcond */ | ||
| 198 | { | 211 | { | 
| 199 | private: | 212 | private: | 
| 200 | typedef struct ListLink<value> Link; | 213 | typedef struct ListLink<value> Link; | 
| @@ -202,9 +215,9 @@ namespace Bu | |||
| 202 | typedef struct ListCore<value, valuealloc, linkalloc> Core; | 215 | typedef struct ListCore<value, valuealloc, linkalloc> Core; | 
| 203 | 216 | ||
| 204 | protected: | 217 | protected: | 
| 205 | using SharedCore< Core >::core; | 218 | using SharedCore<MyType, Core>::core; | 
| 206 | using SharedCore< Core >::_hardCopy; | 219 | using SharedCore<MyType, Core>::_hardCopy; | 
| 207 | using SharedCore< Core >::_allocateCore; | 220 | using SharedCore<MyType, Core>::_allocateCore; | 
| 208 | 221 | ||
| 209 | public: | 222 | public: | 
| 210 | struct const_iterator; | 223 | struct const_iterator; | 
| @@ -215,7 +228,7 @@ namespace Bu | |||
| 215 | } | 228 | } | 
| 216 | 229 | ||
| 217 | List( const MyType &src ) : | 230 | List( const MyType &src ) : | 
| 218 | SharedCore< Core >( src ) | 231 | SharedCore<MyType, Core >( src ) | 
| 219 | { | 232 | { | 
| 220 | } | 233 | } | 
| 221 | 234 | ||
| diff --git a/src/listhash.cpp b/src/listhash.cpp deleted file mode 100644 index b798a1f..0000000 --- a/src/listhash.cpp +++ /dev/null | |||
| @@ -1,8 +0,0 @@ | |||
| 1 | /* | ||
| 2 | * Copyright (C) 2007-2010 Xagasoft, All rights reserved. | ||
| 3 | * | ||
| 4 | * This file is part of the libbu++ library and is released under the | ||
| 5 | * terms of the license contained in the file LICENSE. | ||
| 6 | */ | ||
| 7 | |||
| 8 | |||
| diff --git a/src/listhash.h b/src/listhash.h deleted file mode 100644 index e5ec4ee..0000000 --- a/src/listhash.h +++ /dev/null | |||
| @@ -1,54 +0,0 @@ | |||
| 1 | /* | ||
| 2 | * Copyright (C) 2007-2010 Xagasoft, All rights reserved. | ||
| 3 | * | ||
| 4 | * This file is part of the libbu++ library and is released under the | ||
| 5 | * terms of the license contained in the file LICENSE. | ||
| 6 | */ | ||
| 7 | |||
| 8 | #ifndef BU_LIST_HASH_H | ||
| 9 | #define BU_LIST_HASH_H | ||
| 10 | |||
| 11 | #include "bu/hash.h" | ||
| 12 | #include "bu/list.h" | ||
| 13 | |||
| 14 | namespace Bu | ||
| 15 | { | ||
| 16 | template<typename key, typename value, typename sizecalc = __calcNextTSize_fast, typename keyalloc = std::allocator<key>, typename valuealloc = std::allocator<Bu::List<value> >, typename challoc = std::allocator<uint32_t> > | ||
| 17 | class ListHash : public Hash<key, Bu::List<value>, sizecalc, keyalloc, valuealloc, challoc> | ||
| 18 | { | ||
| 19 | typedef Hash<key, Bu::List<value>, sizecalc, keyalloc, valuealloc, challoc> ParentType; | ||
| 20 | public: | ||
| 21 | ListHash() | ||
| 22 | { | ||
| 23 | } | ||
| 24 | |||
| 25 | ListHash( const ListHash &src ) : | ||
| 26 | ParentType( src ) | ||
| 27 | { | ||
| 28 | } | ||
| 29 | |||
| 30 | virtual ~ListHash() | ||
| 31 | { | ||
| 32 | } | ||
| 33 | |||
| 34 | ListHash &operator=( const ListHash &src ) | ||
| 35 | { | ||
| 36 | *dynamic_cast<ParentType *>(this) = | ||
| 37 | dynamic_cast<ParentType &>(src); | ||
| 38 | } | ||
| 39 | |||
| 40 | virtual void insert( const key &k, const value &v ) | ||
| 41 | { | ||
| 42 | if( !has( k ) ) | ||
| 43 | { | ||
| 44 | ParentType::insert( k, List<value>() ); | ||
| 45 | } | ||
| 46 | get( k ).append( v ); | ||
| 47 | } | ||
| 48 | |||
| 49 | private: | ||
| 50 | }; | ||
| 51 | |||
| 52 | }; | ||
| 53 | |||
| 54 | #endif | ||
| diff --git a/src/myriad.h b/src/myriad.h index 3d203bb..582d310 100644 --- a/src/myriad.h +++ b/src/myriad.h | |||
| @@ -46,7 +46,7 @@ namespace Bu | |||
| 46 | * Header format is as follows: | 46 | * Header format is as follows: | 
| 47 | * | 47 | * | 
| 48 | * MMMMvBssssSSSS* | 48 | * MMMMvBssssSSSS* | 
| 49 | * M = Magic number (FFC399BD) | 49 | * M = Magic number (0AD3FA84) | 
| 50 | * v = version number | 50 | * v = version number | 
| 51 | * B = Bits per int | 51 | * B = Bits per int | 
| 52 | * s = Blocksize in bytes | 52 | * s = Blocksize in bytes | 
| diff --git a/src/myriadfs.cpp b/src/myriadfs.cpp new file mode 100644 index 0000000..af60c08 --- /dev/null +++ b/src/myriadfs.cpp | |||
| @@ -0,0 +1,78 @@ | |||
| 1 | #include "bu/myriadfs.h" | ||
| 2 | #include "bu/myriadstream.h" | ||
| 3 | |||
| 4 | #include <string.h> | ||
| 5 | #include <unistd.h> | ||
| 6 | |||
| 7 | namespace Bu { subExceptionDef( MyriadFsException ) } | ||
| 8 | |||
| 9 | #define Myriad_Fs_MAGIC_CODE ((char *)"\xa7\x18\x8b\x39") | ||
| 10 | |||
| 11 | Bu::MyriadFs::MyriadFs( Bu::Stream &rStore, int iBlockSize ) : | ||
| 12 | rStore( rStore ), | ||
| 13 | mStore( rStore, iBlockSize ) | ||
| 14 | { | ||
| 15 | if( mStore.hasStream( 1 ) ) | ||
| 16 | { | ||
| 17 | // Check to see if this is a MyriadFs stream. | ||
| 18 | Bu::MyriadStream ms = mStore.openStream( 1 ); | ||
| 19 | char sMagic[4]; | ||
| 20 | if( ms.read( sMagic, 4 ) < 4 ) | ||
| 21 | throw MyriadFsException("The provided stream does not appear to be " | ||
| 22 | "a MyriadFs stream."); | ||
| 23 | if( ::strncmp( sMagic, Myriad_Fs_MAGIC_CODE, 4 ) ) | ||
| 24 | throw MyriadFsException("The provided stream does not appear to be " | ||
| 25 | "a MyriadFs stream."); | ||
| 26 | |||
| 27 | int8_t iVer; | ||
| 28 | } | ||
| 29 | else | ||
| 30 | { | ||
| 31 | // Create initial header stream | ||
| 32 | { | ||
| 33 | mStore.createStream( 1 ); | ||
| 34 | Bu::MyriadStream ms = mStore.openStream( 1 ); | ||
| 35 | ms.write( Myriad_Fs_MAGIC_CODE, 4 ); | ||
| 36 | int8_t iVer = 1; | ||
| 37 | int32_t iTmp = 1; | ||
| 38 | ms.write( &iVer, 1 ); | ||
| 39 | ms.write( &iBlockSize, 4 ); | ||
| 40 | ms.write( &iTmp, 4 ); // iNumNodes | ||
| 41 | iTmp = 0; | ||
| 42 | ms.write( &iTmp, 4 ); // iInode | ||
| 43 | ms.write( &iTmp, 0 ); // iPosition | ||
| 44 | } | ||
| 45 | |||
| 46 | // Create initial inode stream, with one root node. | ||
| 47 | { | ||
| 48 | mStore.createStream( 2 ); | ||
| 49 | Bu::MyriadStream ms = mStore.openStream( 2 ); | ||
| 50 | int32_t iUser = 0, iGroup = 0; | ||
| 51 | #ifndef WIN32 | ||
| 52 | iUser = getuid(); | ||
| 53 | iGroup = getgid(); | ||
| 54 | #endif | ||
| 55 | int32_t iTmp32 = 0; | ||
| 56 | int16_t iTmp16 = 0; | ||
| 57 | ms.write( &iUser, 4 ); | ||
| 58 | ms.write( &iGroup, 4 ); | ||
| 59 | ms.write( &iTmp16, 2 ); | ||
| 60 | ms.write( &iTmp16, 2 ); | ||
| 61 | iTmp32 = 3; | ||
| 62 | ms.write( &iTmp32, 4 ); | ||
| 63 | iTmp32 = 0; | ||
| 64 | ms.write( &iTmp32, 4 ); | ||
| 65 | ms.write( &iTmp16, 2 ); | ||
| 66 | } | ||
| 67 | |||
| 68 | // Create inode 0's storage stream. | ||
| 69 | { | ||
| 70 | mStore.createStream( 3 ); | ||
| 71 | } | ||
| 72 | } | ||
| 73 | } | ||
| 74 | |||
| 75 | Bu::MyriadFs::~MyriadFs() | ||
| 76 | { | ||
| 77 | } | ||
| 78 | |||
| diff --git a/src/myriadfs.h b/src/myriadfs.h new file mode 100644 index 0000000..42a3493 --- /dev/null +++ b/src/myriadfs.h | |||
| @@ -0,0 +1,54 @@ | |||
| 1 | #ifndef MYRIAD_FS_H | ||
| 2 | #define MYRIAD_FS_H | ||
| 3 | |||
| 4 | #include "bu/myriad.h" | ||
| 5 | |||
| 6 | namespace Bu | ||
| 7 | { | ||
| 8 | class Stream; | ||
| 9 | |||
| 10 | subExceptionDecl( MyriadFsException ); | ||
| 11 | |||
| 12 | /** | ||
| 13 | * A POSIX compliant, node based filesystem built on top of Myriad. | ||
| 14 | * | ||
| 15 | * A header is placed into stream 1. | ||
| 16 | * Header format: | ||
| 17 | * int32_t iMagicHeader (A7188B39) | ||
| 18 | * int8_t iVersion (1) | ||
| 19 | * int32_t iNodeSize | ||
| 20 | * int32_t iNumNodes | ||
| 21 | * NodeLookup[iNumNodes] nNode | ||
| 22 | * | ||
| 23 | * Node lookup: | ||
| 24 | * int32_t iInode | ||
| 25 | * int32_t iPosition | ||
| 26 | * | ||
| 27 | * The node headers or inode structures have a base size of 22 bytes. | ||
| 28 | * Everything else in the block is used for the name. I.e. if you have | ||
| 29 | * a blocksize of 512 bytes then you wind up with a max name size of | ||
| 30 | * 512-22=490 characters, or a blocksize of 256 gives you 234 chraacters | ||
| 31 | * as a max. The node headers are all stored in stream 2. | ||
| 32 | * Basic node header format: | ||
| 33 | * int32_t iUser | ||
| 34 | * int32_t iGroup | ||
| 35 | * int16_t iMeta | ||
| 36 | * int16_t iPerms | ||
| 37 | * int32_t iStreamIndex | ||
| 38 | * int32_t iParentNode | ||
| 39 | * int16_t iNameSize | ||
| 40 | * char[iNameSize] sName | ||
| 41 | */ | ||
| 42 | class MyriadFs | ||
| 43 | { | ||
| 44 | public: | ||
| 45 | MyriadFs( Bu::Stream &rStore, int iBlockSize=512 ); | ||
| 46 | virtual ~MyriadFs(); | ||
| 47 | |||
| 48 | private: | ||
| 49 | Bu::Stream &rStore; | ||
| 50 | Bu::Myriad mStore; | ||
| 51 | }; | ||
| 52 | }; | ||
| 53 | |||
| 54 | #endif | ||
| diff --git a/src/newline.h b/src/newline.h index b69cdb5..243c876 100644 --- a/src/newline.h +++ b/src/newline.h | |||
| @@ -14,7 +14,7 @@ namespace Bu | |||
| 14 | { | 14 | { | 
| 15 | /** | 15 | /** | 
| 16 | * Converts new-line characters from any standard convention into linefeeds | 16 | * Converts new-line characters from any standard convention into linefeeds | 
| 17 | * (\n) on reading, and converts them to either your OS's standard or a | 17 | * (\\n) on reading, and converts them to either your OS's standard or a | 
| 18 | * specified standard, depending on how you construct the class. | 18 | * specified standard, depending on how you construct the class. | 
| 19 | * | 19 | * | 
| 20 | * If you're reading in a text file, then this filter is practically | 20 | * If you're reading in a text file, then this filter is practically | 
| diff --git a/src/paramproc.cpp b/src/paramproc.cpp deleted file mode 100644 index f4fd36e..0000000 --- a/src/paramproc.cpp +++ /dev/null | |||
| @@ -1,523 +0,0 @@ | |||
| 1 | /* | ||
| 2 | * Copyright (C) 2007-2010 Xagasoft, All rights reserved. | ||
| 3 | * | ||
| 4 | * This file is part of the libbu++ library and is released under the | ||
| 5 | * terms of the license contained in the file LICENSE. | ||
| 6 | */ | ||
| 7 | |||
| 8 | #include "bu/paramproc.h" | ||
| 9 | #include <stdio.h> | ||
| 10 | #include <stdlib.h> | ||
| 11 | #include <string.h> | ||
| 12 | |||
| 13 | #define ptrtype( iitype, iiname ) \ | ||
| 14 | Bu::ParamProc::ParamPtr::ParamPtr( iitype *iiname ) : \ | ||
| 15 | type( vt ##iiname ) { val.iiname = iiname; } | ||
| 16 | |||
| 17 | Bu::ParamProc::ParamPtr::ParamPtr() | ||
| 18 | { | ||
| 19 | val.str = NULL; | ||
| 20 | type = vtunset; | ||
| 21 | } | ||
| 22 | |||
| 23 | ptrtype( Bu::FString, str ); | ||
| 24 | ptrtype( uint64_t, uint64 ); | ||
| 25 | ptrtype( uint32_t, uint32 ); | ||
| 26 | ptrtype( uint16_t, uint16 ); | ||
| 27 | ptrtype( uint8_t, uint8 ); | ||
| 28 | ptrtype( int64_t, int64 ); | ||
| 29 | ptrtype( int32_t, int32 ); | ||
| 30 | ptrtype( int16_t, int16 ); | ||
| 31 | ptrtype( int8_t, int8 ); | ||
| 32 | ptrtype( float, float32 ); | ||
| 33 | ptrtype( double, float64 ); | ||
| 34 | ptrtype( long double, float96 ); | ||
| 35 | ptrtype( bool, bln ); | ||
| 36 | |||
| 37 | Bu::ParamProc::ParamPtr &Bu::ParamProc::ParamPtr::operator=( ParamProc::ParamPtr &ptr ) | ||
| 38 | { | ||
| 39 | val = ptr.val; | ||
| 40 | type = ptr.type; | ||
| 41 | |||
| 42 | return *this; | ||
| 43 | } | ||
| 44 | |||
| 45 | bool Bu::ParamProc::ParamPtr::isSet() | ||
| 46 | { | ||
| 47 | return type != vtunset; | ||
| 48 | } | ||
| 49 | |||
| 50 | Bu::ParamProc::ParamPtr &Bu::ParamProc::ParamPtr::operator=( const char *str ) | ||
| 51 | { | ||
| 52 | if( !isSet() ) return *this; | ||
| 53 | switch( type ) | ||
| 54 | { | ||
| 55 | case vtstr: | ||
| 56 | (*val.str) = str; | ||
| 57 | break; | ||
| 58 | |||
| 59 | case vtuint64: | ||
| 60 | (*val.uint64) = strtoull( str, NULL, 10 ); | ||
| 61 | break; | ||
| 62 | |||
| 63 | case vtuint32: | ||
| 64 | (*val.uint32) = strtoul( str, NULL, 10 ); | ||
| 65 | break; | ||
| 66 | |||
| 67 | case vtuint16: | ||
| 68 | (*val.uint16) = (uint16_t)strtoul( str, NULL, 10 ); | ||
| 69 | break; | ||
| 70 | |||
| 71 | case vtuint8: | ||
| 72 | (*val.uint8) = (uint8_t)strtoul( str, NULL, 10 ); | ||
| 73 | break; | ||
| 74 | |||
| 75 | case vtint64: | ||
| 76 | (*val.int64) = strtoll( str, NULL, 10 ); | ||
| 77 | break; | ||
| 78 | |||
| 79 | case vtint32: | ||
| 80 | (*val.int32) = strtol( str, NULL, 10 ); | ||
| 81 | break; | ||
| 82 | |||
| 83 | case vtint16: | ||
| 84 | (*val.int16) = (int16_t)strtol( str, NULL, 10 ); | ||
| 85 | break; | ||
| 86 | |||
| 87 | case vtint8: | ||
| 88 | (*val.int8) = (int8_t)strtol( str, NULL, 10 ); | ||
| 89 | break; | ||
| 90 | |||
| 91 | case vtfloat32: | ||
| 92 | (*val.float32) = strtof( str, NULL ); | ||
| 93 | break; | ||
| 94 | |||
| 95 | case vtfloat64: | ||
| 96 | (*val.float64) = strtod( str, NULL ); | ||
| 97 | break; | ||
| 98 | |||
| 99 | case vtfloat96: | ||
| 100 | (*val.float96) = strtold( str, NULL ); | ||
| 101 | break; | ||
| 102 | |||
| 103 | case vtbln: | ||
| 104 | if( strcasecmp("yes", str ) == 0 || | ||
| 105 | strcasecmp("true", str ) == 0 ) | ||
| 106 | { | ||
| 107 | (*val.bln) = true; | ||
| 108 | } | ||
| 109 | else | ||
| 110 | { | ||
| 111 | (*val.bln) = false; | ||
| 112 | } | ||
| 113 | break; | ||
| 114 | } | ||
| 115 | |||
| 116 | return *this; | ||
| 117 | } | ||
| 118 | |||
| 119 | Bu::ParamProc::ParamProc() | ||
| 120 | { | ||
| 121 | } | ||
| 122 | |||
| 123 | Bu::ParamProc::~ParamProc() | ||
| 124 | { | ||
| 125 | for( Bu::List<ArgSpec *>::iterator i = lArg.begin(); | ||
| 126 | i != lArg.end(); i++ ) | ||
| 127 | { | ||
| 128 | delete *i; | ||
| 129 | } | ||
| 130 | |||
| 131 | for( Bu::List<Banner *>::iterator i = lBan.begin(); | ||
| 132 | i != lBan.end(); i++ ) | ||
| 133 | { | ||
| 134 | delete *i; | ||
| 135 | } | ||
| 136 | |||
| 137 | } | ||
| 138 | /* | ||
| 139 | void Bu::ParamProc::addParam( const char *lpWord, char cChar, Proc proc, ParamPtr val ) | ||
| 140 | { | ||
| 141 | printf("Calling callback...\n"); | ||
| 142 | val = "Hello there, this is set in the ParamProc"; | ||
| 143 | (this->*proc)(); | ||
| 144 | }*/ | ||
| 145 | |||
| 146 | void Bu::ParamProc::addParam( const char *lpWord, char cChar, Proc proc, | ||
| 147 | ParamPtr val, const char *lpDesc, const char *lpExtra, | ||
| 148 | const char *lpValue ) | ||
| 149 | { | ||
| 150 | ArgSpec *as = new ArgSpec; | ||
| 151 | if( lpWord ) | ||
| 152 | as->sWord = lpWord; | ||
| 153 | |||
| 154 | as->cChar = cChar; | ||
| 155 | as->proc = proc; | ||
| 156 | as->val = val; | ||
| 157 | if( lpDesc ) | ||
| 158 | as->sDesc = lpDesc; | ||
| 159 | if( lpExtra ) | ||
| 160 | as->sExtra = lpExtra; | ||
| 161 | if( lpValue ) | ||
| 162 | as->sValue = lpValue; | ||
| 163 | |||
| 164 | lArg.append( as ); | ||
| 165 | |||
| 166 | if( !lBan.isEmpty() ) | ||
| 167 | { | ||
| 168 | if( lBan.last()->pBefore == NULL ) | ||
| 169 | lBan.last()->pBefore = as; | ||
| 170 | } | ||
| 171 | } | ||
| 172 | |||
| 173 | void Bu::ParamProc::addParam( const char *lpWord, char cChar, Proc proc, | ||
| 174 | const char *lpDesc, const char *lpExtra, | ||
| 175 | const char *lpValue ) | ||
| 176 | { | ||
| 177 | addParam( lpWord, cChar, proc, ParamPtr(), lpDesc, lpExtra, lpValue ); | ||
| 178 | } | ||
| 179 | |||
| 180 | void Bu::ParamProc::addParam( const char *lpWord, char cChar, ParamPtr val, | ||
| 181 | const char *lpDesc, const char *lpExtra, | ||
| 182 | const char *lpValue ) | ||
| 183 | { | ||
| 184 | addParam( lpWord, cChar, NULL, val, lpDesc, lpExtra, lpValue ); | ||
| 185 | } | ||
| 186 | |||
| 187 | void Bu::ParamProc::addParam( const char *lpWord, Proc proc, ParamPtr val, | ||
| 188 | const char *lpDesc, const char *lpExtra, | ||
| 189 | const char *lpValue ) | ||
| 190 | { | ||
| 191 | addParam( lpWord, '\0', proc, val, lpDesc, lpExtra, lpValue ); | ||
| 192 | } | ||
| 193 | |||
| 194 | void Bu::ParamProc::addParam( const char *lpWord, Proc proc, | ||
| 195 | const char *lpDesc, const char *lpExtra, | ||
| 196 | const char *lpValue ) | ||
| 197 | { | ||
| 198 | addParam( lpWord, '\0', proc, ParamPtr(), lpDesc, lpExtra, lpValue ); | ||
| 199 | } | ||
| 200 | |||
| 201 | void Bu::ParamProc::addParam( const char *lpWord, ParamPtr val, | ||
| 202 | const char *lpDesc, const char *lpExtra, | ||
| 203 | const char *lpValue ) | ||
| 204 | { | ||
| 205 | addParam( lpWord, '\0', NULL, val, lpDesc, lpExtra, lpValue ); | ||
| 206 | } | ||
| 207 | |||
| 208 | void Bu::ParamProc::addParam( char cChar, Proc proc, ParamPtr val, | ||
| 209 | const char *lpDesc, const char *lpExtra, | ||
| 210 | const char *lpValue ) | ||
| 211 | { | ||
| 212 | addParam( NULL, cChar, proc, val, lpDesc, lpExtra, lpValue ); | ||
| 213 | } | ||
| 214 | |||
| 215 | void Bu::ParamProc::addParam( char cChar, Proc proc, | ||
| 216 | const char *lpDesc, const char *lpExtra, | ||
| 217 | const char *lpValue ) | ||
| 218 | { | ||
| 219 | addParam( NULL, cChar, proc, ParamPtr(), lpDesc, lpExtra, lpValue ); | ||
| 220 | } | ||
| 221 | |||
| 222 | void Bu::ParamProc::addParam( char cChar, ParamPtr val, | ||
| 223 | const char *lpDesc, const char *lpExtra, | ||
| 224 | const char *lpValue ) | ||
| 225 | { | ||
| 226 | addParam( NULL, cChar, NULL, val, lpDesc, lpExtra, lpValue ); | ||
| 227 | } | ||
| 228 | |||
| 229 | void Bu::ParamProc::process( int argc, char *argv[] ) | ||
| 230 | { | ||
| 231 | for( int arg = 1; arg < argc; arg++ ) | ||
| 232 | { | ||
| 233 | //printf(":::%d:::%s\n", arg, argv[arg] ); | ||
| 234 | if( argv[arg][0] == '-' ) | ||
| 235 | { | ||
| 236 | if( argv[arg][1] == '-' ) | ||
| 237 | { | ||
| 238 | ArgSpec *s = checkWord( argv[arg]+2 ); | ||
| 239 | if( s ) | ||
| 240 | { | ||
| 241 | if( argv[arg][s->sWord.getSize()+2] == '=' ) | ||
| 242 | { | ||
| 243 | if( s->val.isSet() ) | ||
| 244 | { | ||
| 245 | if( s->sValue == "" ) | ||
| 246 | { | ||
| 247 | s->val = argv[arg]+s->sWord.getSize()+3; | ||
| 248 | } | ||
| 249 | else | ||
| 250 | { | ||
| 251 | s->val = s->sValue.getStr(); | ||
| 252 | } | ||
| 253 | } | ||
| 254 | if( s->proc ) | ||
| 255 | { | ||
| 256 | char **tmp = new char*[argc-arg]; | ||
| 257 | tmp[0] = argv[arg]+s->sWord.getSize()+3; | ||
| 258 | for( int k = 1; k < argc-arg; k++ ) | ||
| 259 | tmp[k] = argv[arg+k]; | ||
| 260 | int ret = (this->*s->proc)( argc-arg, tmp ); | ||
| 261 | if( ret > 0 ) | ||
| 262 | { | ||
| 263 | arg += ret-1; | ||
| 264 | } | ||
| 265 | delete tmp; | ||
| 266 | } | ||
| 267 | } | ||
| 268 | else | ||
| 269 | { | ||
| 270 | int add = 0; | ||
| 271 | if( s->val.isSet() ) | ||
| 272 | { | ||
| 273 | if( s->sValue == "" ) | ||
| 274 | { | ||
| 275 | if( arg+1 >= argc ) | ||
| 276 | { | ||
| 277 | return; | ||
| 278 | } | ||
| 279 | s->val = argv[arg+1]; | ||
| 280 | add++; | ||
| 281 | } | ||
| 282 | else | ||
| 283 | { | ||
| 284 | s->val = s->sValue.getStr(); | ||
| 285 | } | ||
| 286 | } | ||
| 287 | if( s->proc ) | ||
| 288 | { | ||
| 289 | int ret = (this->*s->proc)( | ||
| 290 | argc-arg-1, argv+arg+1 ); | ||
| 291 | |||
| 292 | if( ret > add ) | ||
| 293 | add = 0; | ||
| 294 | else | ||
| 295 | add -= ret; | ||
| 296 | arg += ret; | ||
| 297 | } | ||
| 298 | arg += add; | ||
| 299 | } | ||
| 300 | continue; | ||
| 301 | } | ||
| 302 | else | ||
| 303 | { | ||
| 304 | unknownParam( argc-arg, argv+arg ); | ||
| 305 | } | ||
| 306 | } | ||
| 307 | else | ||
| 308 | { | ||
| 309 | for( int chr = 1; argv[arg][chr]; chr++ ) | ||
| 310 | { | ||
| 311 | ArgSpec *s = checkLetr( argv[arg][chr] ); | ||
| 312 | if( s ) | ||
| 313 | { | ||
| 314 | if( argv[arg][chr+1] != '\0' ) | ||
| 315 | { | ||
| 316 | bool bUsed = false; | ||
| 317 | if( s->val.isSet() ) | ||
| 318 | { | ||
| 319 | if( s->sValue == "" ) | ||
| 320 | { | ||
| 321 | s->val = argv[arg]+chr+1; | ||
| 322 | bUsed = true; | ||
| 323 | } | ||
| 324 | else | ||
| 325 | { | ||
| 326 | s->val = s->sValue.getStr(); | ||
| 327 | } | ||
| 328 | } | ||
| 329 | if( s->proc ) | ||
| 330 | { | ||
| 331 | char **tmp = new char*[argc-arg]; | ||
| 332 | tmp[0] = argv[arg]+chr+1; | ||
| 333 | for( int k = 1; k < argc-arg; k++ ) | ||
| 334 | tmp[k] = argv[arg+k]; | ||
| 335 | int ret = (this->*s->proc)( argc-arg, tmp ); | ||
| 336 | if( ret > 0 ) | ||
| 337 | { | ||
| 338 | arg += ret - 1; | ||
| 339 | delete tmp; | ||
| 340 | break; | ||
| 341 | } | ||
| 342 | delete tmp; | ||
| 343 | } | ||
| 344 | if( bUsed ) | ||
| 345 | { | ||
| 346 | break; | ||
| 347 | } | ||
| 348 | } | ||
| 349 | else | ||
| 350 | { | ||
| 351 | bool bUsed = false; | ||
| 352 | if( s->val.isSet() ) | ||
| 353 | { | ||
| 354 | if( s->sValue == "" ) | ||
| 355 | { | ||
| 356 | s->val = argv[arg+1]; | ||
| 357 | bUsed = true; | ||
| 358 | } | ||
| 359 | else | ||
| 360 | { | ||
| 361 | s->val = s->sValue.getStr(); | ||
| 362 | } | ||
| 363 | } | ||
| 364 | if( s->proc ) | ||
| 365 | { | ||
| 366 | int ret = (this->*s->proc)( | ||
| 367 | argc-arg-1, argv+arg+1 | ||
| 368 | ); | ||
| 369 | if( ret > 0 ) | ||
| 370 | { | ||
| 371 | arg += ret; | ||
| 372 | break; | ||
| 373 | } | ||
| 374 | } | ||
| 375 | if( bUsed ) | ||
| 376 | { | ||
| 377 | arg++; | ||
| 378 | break; | ||
| 379 | } | ||
| 380 | } | ||
| 381 | } | ||
| 382 | else | ||
| 383 | { | ||
| 384 | unknownParam( argc-arg, argv+arg ); | ||
| 385 | } | ||
| 386 | } | ||
| 387 | } | ||
| 388 | } | ||
| 389 | else | ||
| 390 | { | ||
| 391 | cmdParam( argc-arg, argv+arg ); | ||
| 392 | } | ||
| 393 | } | ||
| 394 | } | ||
| 395 | |||
| 396 | Bu::ParamProc::ArgSpec *Bu::ParamProc::checkWord( const char *arg ) | ||
| 397 | { | ||
| 398 | //printf("Checking \"%s\"...\n", arg ); | ||
| 399 | Bu::List<ArgSpec *>::const_iterator i = lArg.begin(); | ||
| 400 | for( ; i != lArg.end(); i++ ) | ||
| 401 | { | ||
| 402 | if( (*i)->sWord == "" ) | ||
| 403 | continue; | ||
| 404 | |||
| 405 | if( !strcmp( (*i)->sWord.getStr(), arg ) ) | ||
| 406 | return *i; | ||
| 407 | |||
| 408 | if( (*i)->val.isSet() ) | ||
| 409 | { | ||
| 410 | if( !strncmp( (*i)->sWord.getStr(), arg, (*i)->sWord.getSize() ) && | ||
| 411 | arg[(*i)->sWord.getSize()] == '=' ) | ||
| 412 | { | ||
| 413 | return *i; | ||
| 414 | } | ||
| 415 | } | ||
| 416 | } | ||
| 417 | |||
| 418 | return NULL; | ||
| 419 | } | ||
| 420 | |||
| 421 | Bu::ParamProc::ArgSpec *Bu::ParamProc::checkLetr( const char arg ) | ||
| 422 | { | ||
| 423 | //printf("Checking \'%c\'...\n", arg ); | ||
| 424 | Bu::List<ArgSpec *>::const_iterator i = lArg.begin(); | ||
| 425 | for( ; i != lArg.end(); i++ ) | ||
| 426 | { | ||
| 427 | if( (*i)->cChar == '\0' ) | ||
| 428 | continue; | ||
| 429 | |||
| 430 | if( (*i)->cChar == arg ) | ||
| 431 | { | ||
| 432 | return *i; | ||
| 433 | } | ||
| 434 | } | ||
| 435 | |||
| 436 | return NULL; | ||
| 437 | } | ||
| 438 | |||
| 439 | int Bu::ParamProc::cmdParam( int /*argc*/, char *argv[] ) | ||
| 440 | { | ||
| 441 | printf("Unhandled command parameter \"%s\" found!\n", argv[0] ); | ||
| 442 | return 0; | ||
| 443 | } | ||
| 444 | |||
| 445 | int Bu::ParamProc::unknownParam( int /*argc*/, char *argv[] ) | ||
| 446 | { | ||
| 447 | printf("Unknown parameter \"%s\" found!\n", argv[0] ); | ||
| 448 | return 0; | ||
| 449 | } | ||
| 450 | |||
| 451 | int Bu::ParamProc::help( int /*argc*/, char * /*argv*/ [] ) | ||
| 452 | { | ||
| 453 | Bu::List<Banner *>::const_iterator b = lBan.begin(); | ||
| 454 | Bu::List<ArgSpec *>::const_iterator i = lArg.begin(); | ||
| 455 | int len=0; | ||
| 456 | for( ; i != lArg.end(); i++ ) | ||
| 457 | { | ||
| 458 | if( len < (*i)->sWord.getSize() + (*i)->sExtra.getSize() ) | ||
| 459 | len = (*i)->sWord.getSize() + (*i)->sExtra.getSize(); | ||
| 460 | } | ||
| 461 | char fmt[10]; | ||
| 462 | sprintf( fmt, "%%-%ds ", len ); | ||
| 463 | |||
| 464 | for( i = lArg.begin(); i != lArg.end(); i++ ) | ||
| 465 | { | ||
| 466 | if( b != lBan.end() ) | ||
| 467 | { | ||
| 468 | if( (*b)->pBefore == (*i) ) | ||
| 469 | { | ||
| 470 | printf( (*b)->sBanner.getStr() ); | ||
| 471 | b++; | ||
| 472 | } | ||
| 473 | } | ||
| 474 | printf(" "); | ||
| 475 | if( (*i)->cChar ) | ||
| 476 | { | ||
| 477 | if( (*i)->sWord.getStr() ) | ||
| 478 | { | ||
| 479 | printf("-%c, ", (*i)->cChar ); | ||
| 480 | } | ||
| 481 | else | ||
| 482 | { | ||
| 483 | printf("-%c ", (*i)->cChar ); | ||
| 484 | } | ||
| 485 | } | ||
| 486 | else | ||
| 487 | { | ||
| 488 | printf(" "); | ||
| 489 | } | ||
| 490 | if( (*i)->sWord.getStr() ) | ||
| 491 | { | ||
| 492 | printf("--"); | ||
| 493 | Bu::FString sTmp = (*i)->sWord.getStr(); | ||
| 494 | if( (*i)->sExtra.getStr() ) | ||
| 495 | sTmp += (*i)->sExtra.getStr(); | ||
| 496 | printf( fmt, sTmp.getStr() ); | ||
| 497 | } | ||
| 498 | else | ||
| 499 | { | ||
| 500 | printf(" "); | ||
| 501 | printf(fmt, "" ); | ||
| 502 | } | ||
| 503 | printf("%s\n", (*i)->sDesc.getStr() ); | ||
| 504 | } | ||
| 505 | if( b != lBan.end() ) | ||
| 506 | { | ||
| 507 | if( (*b)->pBefore == NULL ) | ||
| 508 | { | ||
| 509 | printf( (*b)->sBanner.getStr() ); | ||
| 510 | } | ||
| 511 | } | ||
| 512 | |||
| 513 | exit( 0 ); | ||
| 514 | } | ||
| 515 | |||
| 516 | void Bu::ParamProc::addHelpBanner( const char *sHelpBanner ) | ||
| 517 | { | ||
| 518 | Banner *pBan = new Banner; | ||
| 519 | pBan->sBanner = sHelpBanner; | ||
| 520 | pBan->pBefore = NULL; | ||
| 521 | lBan.append( pBan ); | ||
| 522 | } | ||
| 523 | |||
| diff --git a/src/paramproc.h b/src/paramproc.h deleted file mode 100644 index ddc1876..0000000 --- a/src/paramproc.h +++ /dev/null | |||
| @@ -1,163 +0,0 @@ | |||
| 1 | /* | ||
| 2 | * Copyright (C) 2007-2010 Xagasoft, All rights reserved. | ||
| 3 | * | ||
| 4 | * This file is part of the libbu++ library and is released under the | ||
| 5 | * terms of the license contained in the file LICENSE. | ||
| 6 | */ | ||
| 7 | |||
| 8 | #ifndef BU_PARAM_PROC_H | ||
| 9 | #define BU_PARAM_PROC_H | ||
| 10 | |||
| 11 | #include <stdint.h> | ||
| 12 | #include "bu/list.h" | ||
| 13 | #include "bu/fstring.h" | ||
| 14 | |||
| 15 | namespace Bu | ||
| 16 | { | ||
| 17 | class ParamProc | ||
| 18 | { | ||
| 19 | public: | ||
| 20 | class ParamPtr | ||
| 21 | { | ||
| 22 | public: | ||
| 23 | ParamPtr(); | ||
| 24 | ParamPtr( Bu::FString *str ); | ||
| 25 | ParamPtr( uint64_t *uint64 ); | ||
| 26 | ParamPtr( uint32_t *uint32 ); | ||
| 27 | ParamPtr( uint16_t *uint16 ); | ||
| 28 | ParamPtr( uint8_t *uint8 ); | ||
| 29 | ParamPtr( int64_t *int64 ); | ||
| 30 | ParamPtr( int32_t *int32 ); | ||
| 31 | ParamPtr( int16_t *int16 ); | ||
| 32 | ParamPtr( int8_t *int8 ); | ||
| 33 | ParamPtr( float *float32 ); | ||
| 34 | ParamPtr( double *float64 ); | ||
| 35 | ParamPtr( long double *float96 ); | ||
| 36 | ParamPtr( bool *bln ); | ||
| 37 | |||
| 38 | enum | ||
| 39 | { | ||
| 40 | vtunset, | ||
| 41 | vtstr, | ||
| 42 | vtuint64, | ||
| 43 | vtuint32, | ||
| 44 | vtuint16, | ||
| 45 | vtuint8, | ||
| 46 | vtint64, | ||
| 47 | vtint32, | ||
| 48 | vtint16, | ||
| 49 | vtint8, | ||
| 50 | vtfloat32, | ||
| 51 | vtfloat64, | ||
| 52 | vtfloat96, | ||
| 53 | vtbln, | ||
| 54 | }; | ||
| 55 | ParamPtr &operator=( ParamPtr &ptr ); | ||
| 56 | ParamPtr &operator=( const char *str ); | ||
| 57 | |||
| 58 | bool isSet(); | ||
| 59 | |||
| 60 | private: | ||
| 61 | int type; | ||
| 62 | union | ||
| 63 | { | ||
| 64 | Bu::FString *str; | ||
| 65 | uint64_t *uint64; | ||
| 66 | uint32_t *uint32; | ||
| 67 | uint16_t *uint16; | ||
| 68 | uint8_t *uint8; | ||
| 69 | int64_t *int64; | ||
| 70 | int32_t *int32; | ||
| 71 | int16_t *int16; | ||
| 72 | int8_t *int8; | ||
| 73 | float *float32; | ||
| 74 | double *float64; | ||
| 75 | long double *float96; | ||
| 76 | bool *bln; | ||
| 77 | } val; | ||
| 78 | }; | ||
| 79 | |||
| 80 | typedef int (ParamProc::*Proc)( int, char *[] ); | ||
| 81 | |||
| 82 | typedef struct ArgSpec | ||
| 83 | { | ||
| 84 | uint8_t nFlags; | ||
| 85 | Bu::FString sWord; | ||
| 86 | char cChar; | ||
| 87 | Proc proc; | ||
| 88 | ParamProc::ParamPtr val; | ||
| 89 | Bu::FString sExtra; | ||
| 90 | Bu::FString sDesc; | ||
| 91 | Bu::FString sValue; | ||
| 92 | } ArgSpec; | ||
| 93 | |||
| 94 | public: | ||
| 95 | DEPRECATED | ||
| 96 | ParamProc(); | ||
| 97 | virtual ~ParamProc(); | ||
| 98 | |||
| 99 | void addParam( const char *lpWord, char cChar, Proc proc, ParamPtr val, | ||
| 100 | const char *lpDesc=NULL, const char *lpExtra=NULL, | ||
| 101 | const char *lpValue=NULL | ||
| 102 | ); | ||
| 103 | void addParam( const char *lpWord, char cChar, Proc proc, | ||
| 104 | const char *lpDesc=NULL, const char *lpExtra=NULL, | ||
| 105 | const char *lpValue=NULL | ||
| 106 | ); | ||
| 107 | void addParam( const char *lpWord, char cChar, ParamPtr val, | ||
| 108 | const char *lpDesc=NULL, const char *lpExtra=NULL, | ||
| 109 | const char *lpValue=NULL | ||
| 110 | ); | ||
| 111 | |||
| 112 | void addParam( const char *lpWord, Proc proc, ParamPtr val, | ||
| 113 | const char *lpDesc=NULL, const char *lpExtra=NULL, | ||
| 114 | const char *lpValue=NULL | ||
| 115 | ); | ||
| 116 | void addParam( const char *lpWord, Proc proc, | ||
| 117 | const char *lpDesc=NULL, const char *lpExtra=NULL, | ||
| 118 | const char *lpValue=NULL | ||
| 119 | ); | ||
| 120 | void addParam( const char *lpWord, ParamPtr val, | ||
| 121 | const char *lpDesc=NULL, const char *lpExtra=NULL, | ||
| 122 | const char *lpValue=NULL | ||
| 123 | ); | ||
| 124 | |||
| 125 | void addParam( char cChar, Proc proc, ParamPtr val, | ||
| 126 | const char *lpDesc=NULL, const char *lpExtra=NULL, | ||
| 127 | const char *lpValue=NULL | ||
| 128 | ); | ||
| 129 | void addParam( char cChar, Proc proc, | ||
| 130 | const char *lpDesc=NULL, const char *lpExtra=NULL, | ||
| 131 | const char *lpValue=NULL | ||
| 132 | ); | ||
| 133 | void addParam( char cChar, ParamPtr val, | ||
| 134 | const char *lpDesc=NULL, const char *lpExtra=NULL, | ||
| 135 | const char *lpValue=NULL | ||
| 136 | ); | ||
| 137 | |||
| 138 | void process( int argc, char *argv[] ); | ||
| 139 | void addHelpBanner( const char *sHelpBanner ); | ||
| 140 | |||
| 141 | private: | ||
| 142 | ArgSpec *checkWord( const char *arg ); | ||
| 143 | ArgSpec *checkLetr( const char arg ); | ||
| 144 | |||
| 145 | public: | ||
| 146 | virtual int cmdParam( int argc, char *argv[] ); | ||
| 147 | virtual int unknownParam( int argc, char *argv[] ); | ||
| 148 | virtual int help( int argc, char *argv[] ); | ||
| 149 | |||
| 150 | private: | ||
| 151 | typedef struct Banner | ||
| 152 | { | ||
| 153 | Bu::FString sBanner; | ||
| 154 | ArgSpec *pBefore; | ||
| 155 | } Banner; | ||
| 156 | Bu::List<Banner *> lBan; | ||
| 157 | Bu::List<ArgSpec *> lArg; | ||
| 158 | }; | ||
| 159 | } | ||
| 160 | |||
| 161 | #define mkproc( cls ) static_cast<int (Bu::ParamProc::*)( int, char *[])>(&cls) | ||
| 162 | |||
| 163 | #endif | ||
| diff --git a/src/parser.cpp b/src/parser.cpp index e4e8ae5..4ad4ff9 100644 --- a/src/parser.cpp +++ b/src/parser.cpp | |||
| @@ -22,16 +22,25 @@ void Bu::Parser::popLexer() | |||
| 22 | delete sLexer.peekPop(); | 22 | delete sLexer.peekPop(); | 
| 23 | } | 23 | } | 
| 24 | 24 | ||
| 25 | Lexer::Token *Bu::Parser::popToken() | ||
| 26 | { | ||
| 27 | return sToken.peekPop(); | ||
| 28 | } | ||
| 29 | |||
| 30 | void Bu::Parser::pushToken( Lexer::Token *pTok ) | ||
| 31 | { | ||
| 32 | sToken.push( pTok ); | ||
| 33 | } | ||
| 34 | |||
| 25 | void Bu::Parser::parse() | 35 | void Bu::Parser::parse() | 
| 26 | { | 36 | { | 
| 27 | int iCurNt = iRootNonTerminal; | 37 | int iCurNt = iRootNonTerminal; | 
| 28 | Lexer::Token *ptCur = sLexer.peek()->nextToken(); | 38 | Lexer::Token *ptCur = sLexer.peek()->nextToken(); | 
| 29 | sio << "Token: " << sLexer.peek()->tokenToString( *ptCur ) << sio.nl; | 39 | sio << "Token(a): " << sLexer.peek()->tokenToString( *ptCur ) << sio.nl; | 
| 30 | selectProduction( iCurNt, ptCur ); | 40 | selectProduction( iCurNt, ptCur ); | 
| 31 | 41 | ||
| 32 | while( !sState.isEmpty() ) | 42 | while( !sState.isEmpty() ) | 
| 33 | { | 43 | { | 
| 34 | sio << "Currently: " << *sState.peek() << sio.nl; | ||
| 35 | switch( (*sState.peek()).eType ) | 44 | switch( (*sState.peek()).eType ) | 
| 36 | { | 45 | { | 
| 37 | case State::typeTerminal: | 46 | case State::typeTerminal: | 
| @@ -42,7 +51,7 @@ void Bu::Parser::parse() | |||
| 42 | advanceState(); | 51 | advanceState(); | 
| 43 | delete ptCur; | 52 | delete ptCur; | 
| 44 | ptCur = sLexer.peek()->nextToken(); | 53 | ptCur = sLexer.peek()->nextToken(); | 
| 45 | sio << "Token: " << sLexer.peek()->tokenToString( *ptCur ) << sio.nl; | 54 | sio << "Token(b): " << sLexer.peek()->tokenToString( *ptCur ) << sio.nl; | 
| 46 | } | 55 | } | 
| 47 | else | 56 | else | 
| 48 | { | 57 | { | 
| @@ -59,7 +68,7 @@ void Bu::Parser::parse() | |||
| 59 | sToken.push( ptCur ); | 68 | sToken.push( ptCur ); | 
| 60 | 69 | ||
| 61 | ptCur = sLexer.peek()->nextToken(); | 70 | ptCur = sLexer.peek()->nextToken(); | 
| 62 | sio << "Token: " << sLexer.peek()->tokenToString( *ptCur ) << sio.nl; | 71 | sio << "Token(c): " << sLexer.peek()->tokenToString( *ptCur ) << sio.nl; | 
| 63 | } | 72 | } | 
| 64 | else | 73 | else | 
| 65 | { | 74 | { | 
| @@ -72,7 +81,7 @@ void Bu::Parser::parse() | |||
| 72 | << (*sState.peek()).iIndex << sio.nl; | 81 | << (*sState.peek()).iIndex << sio.nl; | 
| 73 | { | 82 | { | 
| 74 | int iNt = (*sState.peek()).iIndex; | 83 | int iNt = (*sState.peek()).iIndex; | 
| 75 | advanceState(); | 84 | sio << "Current state: " << *sState.peek() << sio.nl; | 
| 76 | if( !selectProduction( iNt, ptCur ) ) | 85 | if( !selectProduction( iNt, ptCur ) ) | 
| 77 | { | 86 | { | 
| 78 | throw Bu::ExceptionBase("Error parsing code."); | 87 | throw Bu::ExceptionBase("Error parsing code."); | 
| @@ -132,7 +141,11 @@ bool Bu::Parser::selectProduction( int iNt, Lexer::Token *ptCur ) | |||
| 132 | } | 141 | } | 
| 133 | } | 142 | } | 
| 134 | if( nt.bCanSkip ) | 143 | if( nt.bCanSkip ) | 
| 144 | { | ||
| 145 | sio << "Nothing matches, skipping non-terminal." << sio.nl; | ||
| 146 | advanceState(); | ||
| 135 | return true; | 147 | return true; | 
| 148 | } | ||
| 136 | sio << "-->(Found nothing)" << sio.nl; | 149 | sio << "-->(Found nothing)" << sio.nl; | 
| 137 | return false; | 150 | return false; | 
| 138 | } | 151 | } | 
| @@ -148,6 +161,7 @@ void Bu::Parser::advanceState() | |||
| 148 | sio.decIndent(); | 161 | sio.decIndent(); | 
| 149 | sState.pop(); | 162 | sState.pop(); | 
| 150 | sio << "State advanced, End of production." << sio.nl; | 163 | sio << "State advanced, End of production." << sio.nl; | 
| 164 | advanceState(); | ||
| 151 | return; | 165 | return; | 
| 152 | } | 166 | } | 
| 153 | sio << "State advanced, now: " << *(sState.peek()) << sio.nl; | 167 | sio << "State advanced, now: " << *(sState.peek()) << sio.nl; | 
| @@ -191,6 +205,11 @@ int Bu::Parser::getNonTerminalId( const Bu::FString &sName ) | |||
| 191 | return hNonTerminalName.get( sName ); | 205 | return hNonTerminalName.get( sName ); | 
| 192 | } | 206 | } | 
| 193 | 207 | ||
| 208 | bool Bu::Parser::hasNonTerminal( const Bu::FString &sName ) | ||
| 209 | { | ||
| 210 | return hNonTerminalName.has( sName ); | ||
| 211 | } | ||
| 212 | |||
| 194 | int Bu::Parser::addReduction( const Bu::FString &sName, const Reduction &r ) | 213 | int Bu::Parser::addReduction( const Bu::FString &sName, const Reduction &r ) | 
| 195 | { | 214 | { | 
| 196 | int iId = aReduction.getSize(); | 215 | int iId = aReduction.getSize(); | 
| @@ -217,6 +236,11 @@ int Bu::Parser::getReductionId( const Bu::FString &sName ) | |||
| 217 | return hReductionName.get( sName ); | 236 | return hReductionName.get( sName ); | 
| 218 | } | 237 | } | 
| 219 | 238 | ||
| 239 | bool Bu::Parser::hasReduction( const Bu::FString &sName ) | ||
| 240 | { | ||
| 241 | return hReductionName.has( sName ); | ||
| 242 | } | ||
| 243 | |||
| 220 | // | 244 | // | 
| 221 | // Bu::Parser::State | 245 | // Bu::Parser::State | 
| 222 | // | 246 | // | 
| diff --git a/src/parser.h b/src/parser.h index 5b5d4a8..a925188 100644 --- a/src/parser.h +++ b/src/parser.h | |||
| @@ -34,6 +34,9 @@ namespace Bu | |||
| 34 | */ | 34 | */ | 
| 35 | void popLexer(); | 35 | void popLexer(); | 
| 36 | 36 | ||
| 37 | Lexer::Token *popToken(); | ||
| 38 | void pushToken( Lexer::Token *pTok ); | ||
| 39 | |||
| 37 | /** | 40 | /** | 
| 38 | * Execute a parse. | 41 | * Execute a parse. | 
| 39 | */ | 42 | */ | 
| @@ -88,11 +91,13 @@ namespace Bu | |||
| 88 | int addNonTerminal( const Bu::FString &sName ); | 91 | int addNonTerminal( const Bu::FString &sName ); | 
| 89 | void setNonTerminal( const Bu::FString &sName, NonTerminal &nt ); | 92 | void setNonTerminal( const Bu::FString &sName, NonTerminal &nt ); | 
| 90 | int getNonTerminalId( const Bu::FString &sName ); | 93 | int getNonTerminalId( const Bu::FString &sName ); | 
| 94 | bool hasNonTerminal( const Bu::FString &sName ); | ||
| 91 | 95 | ||
| 92 | int addReduction( const Bu::FString &sName, const Reduction &r ); | 96 | int addReduction( const Bu::FString &sName, const Reduction &r ); | 
| 93 | int addReduction( const Bu::FString &sName ); | 97 | int addReduction( const Bu::FString &sName ); | 
| 94 | void setReduction( const Bu::FString &sName, const Reduction &r ); | 98 | void setReduction( const Bu::FString &sName, const Reduction &r ); | 
| 95 | int getReductionId( const Bu::FString &sName ); | 99 | int getReductionId( const Bu::FString &sName ); | 
| 100 | bool hasReduction( const Bu::FString &sName ); | ||
| 96 | 101 | ||
| 97 | private: | 102 | private: | 
| 98 | bool selectProduction( int iNt, Lexer::Token *ptCur ); | 103 | bool selectProduction( int iNt, Lexer::Token *ptCur ); | 
| diff --git a/src/ringbuffer.h b/src/ringbuffer.h index e984d18..f4fd58c 100644 --- a/src/ringbuffer.h +++ b/src/ringbuffer.h | |||
| @@ -10,97 +10,218 @@ | |||
| 10 | 10 | ||
| 11 | #include <memory> | 11 | #include <memory> | 
| 12 | #include "bu/exceptionbase.h" | 12 | #include "bu/exceptionbase.h" | 
| 13 | #include "bu/queue.h" | ||
| 14 | #include "bu/sharedcore.h" | ||
| 13 | 15 | ||
| 14 | namespace Bu | 16 | namespace Bu | 
| 15 | { | 17 | { | 
| 16 | /** | 18 | template<typename value, typename valuealloc> class RingBuffer; | 
| 17 | *@ingroup Containers | 19 | |
| 18 | */ | 20 | /** @cond DEVEL */ | 
| 19 | template<typename value, typename valuealloc=std::allocator<value> > | 21 | template<typename value, typename valuealloc> | 
| 20 | class RingBuffer | 22 | class RingBufferCore | 
| 21 | { | 23 | { | 
| 22 | public: | 24 | friend class RingBuffer<value, valuealloc>; | 
| 23 | RingBuffer( int nCapacity ) : | 25 | friend class SharedCore<RingBuffer<value, valuealloc>, | 
| 24 | nCapacity( nCapacity ), | 26 | RingBufferCore<value, valuealloc> >; | 
| 25 | nStart( -1 ), | 27 | private: | 
| 26 | nEnd( -2 ) | 28 | RingBufferCore() : | 
| 29 | iCapacity( 0 ), | ||
| 30 | iStart( -1 ), | ||
| 31 | iEnd( -2 ), | ||
| 32 | aData( NULL ) | ||
| 27 | { | 33 | { | 
| 28 | aData = va.allocate( nCapacity ); | ||
| 29 | } | 34 | } | 
| 30 | 35 | ||
| 31 | virtual ~RingBuffer() | 36 | virtual ~RingBufferCore() | 
| 32 | { | 37 | { | 
| 33 | for( int j = nStart; j < nEnd; j=(j+1%nCapacity) ) | 38 | clear(); | 
| 34 | { | ||
| 35 | va.destroy( &aData[j] ); | ||
| 36 | } | ||
| 37 | va.deallocate( aData, nCapacity ); | ||
| 38 | } | 39 | } | 
| 39 | 40 | ||
| 40 | int getCapacity() | 41 | void init( int iNewCapacity ) | 
| 41 | { | 42 | { | 
| 42 | return nCapacity; | 43 | if( iCapacity > 0 ) | 
| 43 | } | 44 | return; | 
| 44 | 45 | ||
| 45 | bool isFilled() | 46 | iCapacity = iNewCapacity; | 
| 46 | { | 47 | iStart = -1; | 
| 47 | return (nStart == nEnd); | 48 | iEnd = -2; | 
| 49 | aData = va.allocate( iCapacity ); | ||
| 48 | } | 50 | } | 
| 49 | 51 | ||
| 50 | bool isEmpty() | 52 | void clear() | 
| 51 | { | 53 | { | 
| 52 | return (nStart == -1); | 54 | for( int j = iStart; j < iEnd; j=(j+1%iCapacity) ) | 
| 55 | { | ||
| 56 | va.destroy( &aData[j] ); | ||
| 57 | } | ||
| 58 | va.deallocate( aData, iCapacity ); | ||
| 59 | aData = NULL; | ||
| 60 | iCapacity = 0; | ||
| 53 | } | 61 | } | 
| 54 | 62 | ||
| 55 | void enqueue( const value &v ) | 63 | void enqueue( const value &v ) | 
| 56 | { | 64 | { | 
| 57 | if( nStart == -1 ) | 65 | if( iStart == -1 ) | 
| 58 | { | 66 | { | 
| 59 | nStart = 0; | 67 | iStart = 0; | 
| 60 | nEnd = 1; | 68 | iEnd = 1; | 
| 61 | va.construct( &aData[0], v ); | 69 | va.construct( &aData[0], v ); | 
| 62 | } | 70 | } | 
| 63 | else if( nStart == nEnd ) | 71 | else if( iStart == iEnd ) | 
| 64 | { | 72 | { | 
| 65 | throw ExceptionBase("Hey, it's full!"); | 73 | throw ExceptionBase("Hey, it's full!"); | 
| 66 | } | 74 | } | 
| 67 | else | 75 | else | 
| 68 | { | 76 | { | 
| 69 | va.construct( &aData[nEnd], v ); | 77 | va.construct( &aData[iEnd], v ); | 
| 70 | nEnd = (nEnd+1)%nCapacity; | 78 | iEnd = (iEnd+1)%iCapacity; | 
| 71 | } | 79 | } | 
| 72 | } | 80 | } | 
| 73 | 81 | ||
| 74 | value dequeue() | 82 | value dequeue() | 
| 75 | { | 83 | { | 
| 76 | if( nStart == -1 ) | 84 | if( iStart == -1 ) | 
| 77 | { | 85 | { | 
| 78 | throw ExceptionBase("No data"); | 86 | throw ExceptionBase("No data"); | 
| 79 | } | 87 | } | 
| 80 | else | 88 | else | 
| 81 | { | 89 | { | 
| 82 | value &v = aData[nStart]; | 90 | value &v = aData[iStart]; | 
| 83 | va.destroy( &aData[nStart] ); | 91 | va.destroy( &aData[iStart] ); | 
| 84 | nStart = (nStart+1)%nCapacity; | 92 | iStart = (iStart+1)%iCapacity; | 
| 85 | if( nStart == nEnd ) | 93 | if( iStart == iEnd ) | 
| 86 | { | 94 | { | 
| 87 | nStart = -1; | 95 | iStart = -1; | 
| 88 | nEnd = -2; | 96 | iEnd = -2; | 
| 89 | } | 97 | } | 
| 90 | return v; | 98 | return v; | 
| 91 | } | 99 | } | 
| 92 | } | 100 | } | 
| 93 | 101 | ||
| 94 | value &operator[]( int nIndex ) | 102 | value &get( int iIndex ) | 
| 95 | { | 103 | { | 
| 96 | return aData[(nIndex+nStart)%nCapacity]; | 104 | return aData[(iIndex+iStart)%iCapacity]; | 
| 97 | } | 105 | } | 
| 98 | 106 | ||
| 99 | private: | 107 | int getSize() | 
| 100 | int nCapacity; | 108 | { | 
| 109 | if( iStart < 0 ) | ||
| 110 | return 0; | ||
| 111 | if( iEnd == iStart ) | ||
| 112 | return iCapacity; | ||
| 113 | if( iEnd < iStart ) | ||
| 114 | return iEnd-iStart; | ||
| 115 | return iCapacity-(iEnd-iStart); | ||
| 116 | } | ||
| 117 | |||
| 118 | int iCapacity; | ||
| 119 | int iStart, iEnd; | ||
| 101 | value *aData; | 120 | value *aData; | 
| 102 | valuealloc va; | 121 | valuealloc va; | 
| 103 | int nStart, nEnd; | 122 | }; | 
| 123 | /** @endcond */ | ||
| 124 | |||
| 125 | /** | ||
| 126 | *@ingroup Containers | ||
| 127 | */ | ||
| 128 | template<typename value, typename valuealloc=std::allocator<value> > | ||
| 129 | class RingBuffer : public Queue<value>, public SharedCore< | ||
| 130 | RingBuffer<value, valuealloc>, | ||
| 131 | RingBufferCore<value, valuealloc> | ||
| 132 | > | ||
| 133 | { | ||
| 134 | private: | ||
| 135 | typedef RingBuffer<value, valuealloc> MyType; | ||
| 136 | typedef RingBufferCore<value, valuealloc> Core; | ||
| 137 | |||
| 138 | protected: | ||
| 139 | using SharedCore<MyType, Core>::core; | ||
| 140 | using SharedCore<MyType, Core>::_hardCopy; | ||
| 141 | using SharedCore<MyType, Core>::_allocateCore; | ||
| 142 | |||
| 143 | public: | ||
| 144 | RingBuffer( int iCapacity ) | ||
| 145 | { | ||
| 146 | core->init( iCapacity ); | ||
| 147 | } | ||
| 148 | |||
| 149 | RingBuffer( const RingBuffer &rSrc ) : | ||
| 150 | SharedCore<MyType, Core>( rSrc ) | ||
| 151 | { | ||
| 152 | } | ||
| 153 | |||
| 154 | virtual ~RingBuffer() | ||
| 155 | { | ||
| 156 | } | ||
| 157 | |||
| 158 | int getCapacity() const | ||
| 159 | { | ||
| 160 | return core->iCapacity; | ||
| 161 | } | ||
| 162 | |||
| 163 | bool isFilled() const | ||
| 164 | { | ||
| 165 | return (core->iStart == core->iEnd); | ||
| 166 | } | ||
| 167 | |||
| 168 | bool isEmpty() const | ||
| 169 | { | ||
| 170 | return (core->iStart == -1); | ||
| 171 | } | ||
| 172 | |||
| 173 | virtual void enqueue( const value &v ) | ||
| 174 | { | ||
| 175 | _hardCopy(); | ||
| 176 | |||
| 177 | core->enqueue( v ); | ||
| 178 | } | ||
| 179 | |||
| 180 | virtual value dequeue() | ||
| 181 | { | ||
| 182 | _hardCopy(); | ||
| 183 | |||
| 184 | return core->dequeue(); | ||
| 185 | } | ||
| 186 | |||
| 187 | virtual int getSize() const | ||
| 188 | { | ||
| 189 | return core->getSize(); | ||
| 190 | } | ||
| 191 | |||
| 192 | virtual value &peek() | ||
| 193 | { | ||
| 194 | _hardCopy(); | ||
| 195 | |||
| 196 | return core->get( 0 ); | ||
| 197 | } | ||
| 198 | |||
| 199 | virtual const value &peek() const | ||
| 200 | { | ||
| 201 | return core->get( 0 ); | ||
| 202 | } | ||
| 203 | |||
| 204 | value &operator[]( int iIndex ) | ||
| 205 | { | ||
| 206 | _hardCopy(); | ||
| 207 | |||
| 208 | return core->get( iIndex ); | ||
| 209 | } | ||
| 210 | |||
| 211 | protected: | ||
| 212 | virtual Core *_copyCore( Core *src ) | ||
| 213 | { | ||
| 214 | Core *pRet = _allocateCore(); | ||
| 215 | |||
| 216 | pRet->init( src->iCapacity ); | ||
| 217 | int iSize = src->getSize(); | ||
| 218 | for( int j = 0; j < iSize; j++ ) | ||
| 219 | { | ||
| 220 | pRet->enqueue( src->get( j ) ); | ||
| 221 | } | ||
| 222 | |||
| 223 | return pRet; | ||
| 224 | } | ||
| 104 | }; | 225 | }; | 
| 105 | } | 226 | } | 
| 106 | 227 | ||
| diff --git a/src/server.cpp b/src/server.cpp index 51c056a..e701a69 100644 --- a/src/server.cpp +++ b/src/server.cpp | |||
| @@ -8,9 +8,9 @@ | |||
| 8 | #include "bu/server.h" | 8 | #include "bu/server.h" | 
| 9 | #include <errno.h> | 9 | #include <errno.h> | 
| 10 | #include <unistd.h> | 10 | #include <unistd.h> | 
| 11 | #include "bu/serversocket.h" | 11 | #include "bu/tcpserversocket.h" | 
| 12 | #include "bu/client.h" | 12 | #include "bu/client.h" | 
| 13 | #include "bu/socket.h" | 13 | #include "bu/tcpsocket.h" | 
| 14 | #include "bu/config.h" | 14 | #include "bu/config.h" | 
| 15 | 15 | ||
| 16 | Bu::Server::Server() : | 16 | Bu::Server::Server() : | 
| @@ -28,7 +28,7 @@ Bu::Server::~Server() | |||
| 28 | 28 | ||
| 29 | void Bu::Server::addPort( int nPort, int nPoolSize ) | 29 | void Bu::Server::addPort( int nPort, int nPoolSize ) | 
| 30 | { | 30 | { | 
| 31 | ServerSocket *s = new ServerSocket( nPort, nPoolSize ); | 31 | TcpServerSocket *s = new TcpServerSocket( nPort, nPoolSize ); | 
| 32 | int nSocket = s->getSocket(); | 32 | int nSocket = s->getSocket(); | 
| 33 | FD_SET( nSocket, &fdActive ); | 33 | FD_SET( nSocket, &fdActive ); | 
| 34 | hServers.insert( nSocket, s ); | 34 | hServers.insert( nSocket, s ); | 
| @@ -36,7 +36,7 @@ void Bu::Server::addPort( int nPort, int nPoolSize ) | |||
| 36 | 36 | ||
| 37 | void Bu::Server::addPort( const FString &sAddr, int nPort, int nPoolSize ) | 37 | void Bu::Server::addPort( const FString &sAddr, int nPort, int nPoolSize ) | 
| 38 | { | 38 | { | 
| 39 | ServerSocket *s = new ServerSocket( sAddr, nPort, nPoolSize ); | 39 | TcpServerSocket *s = new TcpServerSocket( sAddr, nPort, nPoolSize ); | 
| 40 | int nSocket = s->getSocket(); | 40 | int nSocket = s->getSocket(); | 
| 41 | FD_SET( nSocket, &fdActive ); | 41 | FD_SET( nSocket, &fdActive ); | 
| 42 | hServers.insert( nSocket, s ); | 42 | hServers.insert( nSocket, s ); | 
| @@ -75,7 +75,7 @@ void Bu::Server::scan() | |||
| 75 | { | 75 | { | 
| 76 | if( hServers.has( j ) ) | 76 | if( hServers.has( j ) ) | 
| 77 | { | 77 | { | 
| 78 | ServerSocket *pSrv = hServers.get( j ); | 78 | TcpServerSocket *pSrv = hServers.get( j ); | 
| 79 | addClient( pSrv->accept(), pSrv->getPort() ); | 79 | addClient( pSrv->accept(), pSrv->getPort() ); | 
| 80 | } | 80 | } | 
| 81 | else | 81 | else | 
| @@ -97,7 +97,7 @@ void Bu::Server::scan() | |||
| 97 | { | 97 | { | 
| 98 | pClient->processOutput(); | 98 | pClient->processOutput(); | 
| 99 | } | 99 | } | 
| 100 | catch( Bu::SocketException &e ) | 100 | catch( Bu::TcpSocketException &e ) | 
| 101 | { | 101 | { | 
| 102 | closeClient( j ); | 102 | closeClient( j ); | 
| 103 | } | 103 | } | 
| @@ -136,7 +136,7 @@ void Bu::Server::addClient( int nSocket, int nPort ) | |||
| 136 | FD_SET( nSocket, &fdActive ); | 136 | FD_SET( nSocket, &fdActive ); | 
| 137 | 137 | ||
| 138 | Client *c = new Client( | 138 | Client *c = new Client( | 
| 139 | new Bu::Socket( nSocket ), | 139 | new Bu::TcpSocket( nSocket ), | 
| 140 | new SrvClientLinkFactory() | 140 | new SrvClientLinkFactory() | 
| 141 | ); | 141 | ); | 
| 142 | hClients.insert( nSocket, c ); | 142 | hClients.insert( nSocket, c ); | 
| diff --git a/src/server.h b/src/server.h index 74ee99a..d6726fd 100644 --- a/src/server.h +++ b/src/server.h | |||
| @@ -25,8 +25,8 @@ | |||
| 25 | 25 | ||
| 26 | namespace Bu | 26 | namespace Bu | 
| 27 | { | 27 | { | 
| 28 | class ServerSocket; | 28 | class TcpServerSocket; | 
| 29 | class Socket; | 29 | class TcpSocket; | 
| 30 | class Client; | 30 | class Client; | 
| 31 | 31 | ||
| 32 | /** | 32 | /** | 
| @@ -97,7 +97,7 @@ namespace Bu | |||
| 97 | int nTimeoutSec; | 97 | int nTimeoutSec; | 
| 98 | int nTimeoutUSec; | 98 | int nTimeoutUSec; | 
| 99 | fd_set fdActive; | 99 | fd_set fdActive; | 
| 100 | typedef Hash<int,ServerSocket *> SrvHash; | 100 | typedef Hash<int,TcpServerSocket *> SrvHash; | 
| 101 | SrvHash hServers; | 101 | SrvHash hServers; | 
| 102 | typedef Hash<int,Client *> ClientHash; | 102 | typedef Hash<int,Client *> ClientHash; | 
| 103 | ClientHash hClients; | 103 | ClientHash hClients; | 
| @@ -24,17 +24,10 @@ namespace Bu | |||
| 24 | { | 24 | { | 
| 25 | subExceptionDecl( SetException ) | 25 | subExceptionDecl( SetException ) | 
| 26 | 26 | ||
| 27 | template<typename T> | ||
| 28 | uint32_t __calcHashCode( const T &k ); | ||
| 29 | |||
| 30 | template<typename T> | ||
| 31 | bool __cmpHashKeys( const T &a, const T &b ); | ||
| 32 | |||
| 33 | template<typename key, typename sizecalc = struct __calcNextTSize_fast, typename keyalloc = std::allocator<key>, typename challoc = std::allocator<uint32_t> > | ||
| 34 | class Set; | ||
| 35 | |||
| 36 | /** | 27 | /** | 
| 37 | * Libbu Template Set | 28 | *@todo Set should be rewritten, possibly using a b-tree as ordered storage | 
| 29 | * in the backend. It should use either a b-tree or array for storage and | ||
| 30 | * allow set intersections, unions, etc. | ||
| 38 | *@param key (typename) The datatype of the hashtable keys | 31 | *@param key (typename) The datatype of the hashtable keys | 
| 39 | *@param sizecalc (typename) Functor to compute new table size on rehash | 32 | *@param sizecalc (typename) Functor to compute new table size on rehash | 
| 40 | *@param keyalloc (typename) Memory allocator for hashtable keys | 33 | *@param keyalloc (typename) Memory allocator for hashtable keys | 
| @@ -45,754 +38,7 @@ namespace Bu | |||
| 45 | class Set | 38 | class Set | 
| 46 | { | 39 | { | 
| 47 | public: | 40 | public: | 
| 48 | Set() : | ||
| 49 | nCapacity( 11 ), | ||
| 50 | nFilled( 0 ), | ||
| 51 | nDeleted( 0 ), | ||
| 52 | bFilled( NULL ), | ||
| 53 | bDeleted( NULL ), | ||
| 54 | aKeys( NULL ), | ||
| 55 | aHashCodes( NULL ) | ||
| 56 | { | ||
| 57 | nKeysSize = bitsToBytes( nCapacity ); | ||
| 58 | bFilled = ca.allocate( nKeysSize ); | ||
| 59 | bDeleted = ca.allocate( nKeysSize ); | ||
| 60 | clearBits(); | ||
| 61 | |||
| 62 | aHashCodes = ca.allocate( nCapacity ); | ||
| 63 | aKeys = ka.allocate( nCapacity ); | ||
| 64 | } | ||
| 65 | |||
| 66 | Set( const Set &src ) : | ||
| 67 | nCapacity( src.nCapacity ), | ||
| 68 | nFilled( 0 ), | ||
| 69 | nDeleted( 0 ), | ||
| 70 | bFilled( NULL ), | ||
| 71 | bDeleted( NULL ), | ||
| 72 | aKeys( NULL ), | ||
| 73 | aHashCodes( NULL ) | ||
| 74 | { | ||
| 75 | nKeysSize = bitsToBytes( nCapacity ); | ||
| 76 | bFilled = ca.allocate( nKeysSize ); | ||
| 77 | bDeleted = ca.allocate( nKeysSize ); | ||
| 78 | clearBits(); | ||
| 79 | |||
| 80 | aHashCodes = ca.allocate( nCapacity ); | ||
| 81 | aKeys = ka.allocate( nCapacity ); | ||
| 82 | |||
| 83 | for( uint32_t j = 0; j < src.nCapacity; j++ ) | ||
| 84 | { | ||
| 85 | if( src.isFilled( j ) ) | ||
| 86 | { | ||
| 87 | insert( src.aKeys[j] ); | ||
| 88 | } | ||
| 89 | } | ||
| 90 | } | ||
| 91 | |||
| 92 | /** | ||
| 93 | * Set assignment operator. Clears this hashtable and | ||
| 94 | * copies RH into it. | ||
| 95 | */ | ||
| 96 | Set &operator=( const Set &src ) | ||
| 97 | { | ||
| 98 | for( uint32_t j = 0; j < nCapacity; j++ ) | ||
| 99 | { | ||
| 100 | if( isFilled( j ) ) | ||
| 101 | if( !isDeleted( j ) ) | ||
| 102 | { | ||
| 103 | ka.destroy( &aKeys[j] ); | ||
| 104 | } | ||
| 105 | } | ||
| 106 | ka.deallocate( aKeys, nCapacity ); | ||
| 107 | ca.deallocate( bFilled, nKeysSize ); | ||
| 108 | ca.deallocate( bDeleted, nKeysSize ); | ||
| 109 | ca.deallocate( aHashCodes, nCapacity ); | ||
| 110 | |||
| 111 | nFilled = 0; | ||
| 112 | nDeleted = 0; | ||
| 113 | nCapacity = src.nCapacity; | ||
| 114 | nKeysSize = bitsToBytes( nCapacity ); | ||
| 115 | bFilled = ca.allocate( nKeysSize ); | ||
| 116 | bDeleted = ca.allocate( nKeysSize ); | ||
| 117 | clearBits(); | ||
| 118 | |||
| 119 | aHashCodes = ca.allocate( nCapacity ); | ||
| 120 | aKeys = ka.allocate( nCapacity ); | ||
| 121 | |||
| 122 | for( uint32_t j = 0; j < src.nCapacity; j++ ) | ||
| 123 | { | ||
| 124 | if( src.isFilled( j ) ) | ||
| 125 | { | ||
| 126 | insert( src.aKeys[j] ); | ||
| 127 | } | ||
| 128 | } | ||
| 129 | |||
| 130 | return *this; | ||
| 131 | } | ||
| 132 | |||
| 133 | virtual ~Set() | ||
| 134 | { | ||
| 135 | for( uint32_t j = 0; j < nCapacity; j++ ) | ||
| 136 | { | ||
| 137 | if( isFilled( j ) ) | ||
| 138 | if( !isDeleted( j ) ) | ||
| 139 | { | ||
| 140 | ka.destroy( &aKeys[j] ); | ||
| 141 | } | ||
| 142 | } | ||
| 143 | ka.deallocate( aKeys, nCapacity ); | ||
| 144 | ca.deallocate( bFilled, nKeysSize ); | ||
| 145 | ca.deallocate( bDeleted, nKeysSize ); | ||
| 146 | ca.deallocate( aHashCodes, nCapacity ); | ||
| 147 | } | ||
| 148 | |||
| 149 | /** | ||
| 150 | * Get the current hash table capacity. (Changes at re-hash) | ||
| 151 | *@returns (uint32_t) The current capacity. | ||
| 152 | */ | ||
| 153 | uint32_t getCapacity() | ||
| 154 | { | ||
| 155 | return nCapacity; | ||
| 156 | } | ||
| 157 | |||
| 158 | /** | ||
| 159 | * Get the number of hash locations spoken for. (Including | ||
| 160 | * not-yet-cleaned-up deleted items.) | ||
| 161 | *@returns (uint32_t) The current fill state. | ||
| 162 | */ | ||
| 163 | uint32_t getFill() | ||
| 164 | { | ||
| 165 | return nFilled; | ||
| 166 | } | ||
| 167 | |||
| 168 | /** | ||
| 169 | * Get the number of items stored in the hash table. | ||
| 170 | *@returns (uint32_t) The number of items stored in the hash table. | ||
| 171 | */ | ||
| 172 | uint32_t getSize() | ||
| 173 | { | ||
| 174 | return nFilled-nDeleted; | ||
| 175 | } | ||
| 176 | |||
| 177 | /** | ||
| 178 | * Get the number of items which have been deleted, but not yet | ||
| 179 | * cleaned up. | ||
| 180 | *@returns (uint32_t) The number of deleted items. | ||
| 181 | */ | ||
| 182 | uint32_t getDeleted() | ||
| 183 | { | ||
| 184 | return nDeleted; | ||
| 185 | } | ||
| 186 | |||
| 187 | /** | ||
| 188 | * Insert key (k) into the set | ||
| 189 | *@param k (key_type) Key to list the value under. | ||
| 190 | */ | ||
| 191 | virtual void insert( key k ) | ||
| 192 | { | ||
| 193 | uint32_t hash = __calcHashCode( k ); | ||
| 194 | bool bFill; | ||
| 195 | uint32_t nPos = probe( hash, k, bFill ); | ||
| 196 | |||
| 197 | if( bFill ) | ||
| 198 | { | ||
| 199 | onUpdate(); | ||
| 200 | } | ||
| 201 | else | ||
| 202 | { | ||
| 203 | fill( nPos, k, hash ); | ||
| 204 | onInsert(); | ||
| 205 | } | ||
| 206 | } | ||
| 207 | |||
| 208 | /** | ||
| 209 | * Remove a value from the hash table. | ||
| 210 | *@param k (key_type) The data under this key will be erased. | ||
| 211 | */ | ||
| 212 | virtual void erase( key k ) | ||
| 213 | { | ||
| 214 | uint32_t hash = __calcHashCode( k ); | ||
| 215 | bool bFill; | ||
| 216 | uint32_t nPos = probe( hash, k, bFill ); | ||
| 217 | |||
| 218 | if( bFill ) | ||
| 219 | { | ||
| 220 | _erase( nPos ); | ||
| 221 | onDelete(); | ||
| 222 | } | ||
| 223 | } | ||
| 224 | |||
| 225 | struct iterator; | ||
| 226 | |||
| 227 | /** | ||
| 228 | * Remove a value from the hash pointed to from an iterator. | ||
| 229 | *@param i (iterator &) The data to be erased. | ||
| 230 | */ | ||
| 231 | virtual void erase( struct iterator &i ) | ||
| 232 | { | ||
| 233 | if( this != &i.hsh ) | ||
| 234 | throw SetException("This iterator didn't come from this Hash."); | ||
| 235 | if( isFilled( i.nPos ) && !isDeleted( i.nPos ) ) | ||
| 236 | { | ||
| 237 | _erase( i.nPos ); | ||
| 238 | onDelete(); | ||
| 239 | } | ||
| 240 | } | ||
| 241 | |||
| 242 | /** | ||
| 243 | * Remove all data from the hash table. | ||
| 244 | */ | ||
| 245 | virtual void clear() | ||
| 246 | { | ||
| 247 | for( uint32_t j = 0; j < nCapacity; j++ ) | ||
| 248 | { | ||
| 249 | if( isFilled( j ) ) | ||
| 250 | if( !isDeleted( j ) ) | ||
| 251 | { | ||
| 252 | _erase( j ); | ||
| 253 | onDelete(); | ||
| 254 | } | ||
| 255 | } | ||
| 256 | |||
| 257 | clearBits(); | ||
| 258 | } | ||
| 259 | |||
| 260 | /** | ||
| 261 | * Does the hash table contain an item under key (k). | ||
| 262 | *@param k (key_type) The key to check. | ||
| 263 | *@returns (bool) Whether there was an item in the hash under key (k). | ||
| 264 | */ | ||
| 265 | virtual bool has( key k ) | ||
| 266 | { | ||
| 267 | bool bFill; | ||
| 268 | probe( __calcHashCode( k ), k, bFill, false ); | ||
| 269 | |||
| 270 | return bFill; | ||
| 271 | } | ||
| 272 | |||
| 273 | /** | ||
| 274 | * Iteration structure for iterating through the hash. | ||
| 275 | */ | ||
| 276 | typedef struct iterator | ||
| 277 | { | ||
| 278 | friend class Set<key, sizecalc, keyalloc, challoc>; | ||
| 279 | private: | ||
| 280 | iterator( Set<key, sizecalc, keyalloc, challoc> &hsh ) : | ||
| 281 | hsh( hsh ), | ||
| 282 | nPos( 0 ), | ||
| 283 | bFinished( false ) | ||
| 284 | { | ||
| 285 | nPos = hsh.getFirstPos( bFinished ); | ||
| 286 | } | ||
| 287 | |||
| 288 | iterator( Set<key, sizecalc, keyalloc, challoc> &hsh, bool bDone ) : | ||
| 289 | hsh( hsh ), | ||
| 290 | nPos( 0 ), | ||
| 291 | bFinished( bDone ) | ||
| 292 | { | ||
| 293 | } | ||
| 294 | |||
| 295 | Set<key, sizecalc, keyalloc, challoc> &hsh; | ||
| 296 | uint32_t nPos; | ||
| 297 | bool bFinished; | ||
| 298 | |||
| 299 | public: | ||
| 300 | /** | ||
| 301 | * Iterator incrementation operator. Move the iterator forward. | ||
| 302 | */ | ||
| 303 | iterator operator++( int ) | ||
| 304 | { | ||
| 305 | if( bFinished == false ) | ||
| 306 | nPos = hsh.getNextPos( nPos, bFinished ); | ||
| 307 | |||
| 308 | return *this; | ||
| 309 | } | ||
| 310 | |||
| 311 | /** | ||
| 312 | * Iterator incrementation operator. Move the iterator forward. | ||
| 313 | */ | ||
| 314 | iterator operator++() | ||
| 315 | { | ||
| 316 | if( bFinished == false ) | ||
| 317 | nPos = hsh.getNextPos( nPos, bFinished ); | ||
| 318 | |||
| 319 | return *this; | ||
| 320 | } | ||
| 321 | |||
| 322 | /** | ||
| 323 | * Iterator equality comparison operator. Iterators the same? | ||
| 324 | */ | ||
| 325 | bool operator==( const iterator &oth ) const | ||
| 326 | { | ||
| 327 | if( bFinished != oth.bFinished ) | ||
| 328 | return false; | ||
| 329 | if( bFinished == true ) | ||
| 330 | { | ||
| 331 | return true; | ||
| 332 | } | ||
| 333 | else | ||
| 334 | { | ||
| 335 | if( oth.nPos == nPos ) | ||
| 336 | return true; | ||
| 337 | return false; | ||
| 338 | } | ||
| 339 | } | ||
| 340 | |||
| 341 | /** | ||
| 342 | * Iterator not equality comparison operator. Not the same? | ||
| 343 | */ | ||
| 344 | bool operator!=( const iterator &oth ) const | ||
| 345 | { | ||
| 346 | return !(*this == oth ); | ||
| 347 | } | ||
| 348 | |||
| 349 | /** | ||
| 350 | * Iterator assignment operator. | ||
| 351 | */ | ||
| 352 | iterator operator=( const iterator &oth ) | ||
| 353 | { | ||
| 354 | if( &hsh != &oth.hsh ) | ||
| 355 | throw SetException( | ||
| 356 | "Cannot mix iterators from different set objects."); | ||
| 357 | nPos = oth.nPos; | ||
| 358 | bFinished = oth.bFinished; | ||
| 359 | } | ||
| 360 | |||
| 361 | /** | ||
| 362 | * Iterator dereference operator... err.. get the value | ||
| 363 | *@returns (value_type &) The value behind this iterator. | ||
| 364 | */ | ||
| 365 | key &operator *() | ||
| 366 | { | ||
| 367 | return hsh.getKeyAtPos( nPos ); | ||
| 368 | } | ||
| 369 | |||
| 370 | const key &operator *() const | ||
| 371 | { | ||
| 372 | return hsh.getKeyAtPos( nPos ); | ||
| 373 | } | ||
| 374 | |||
| 375 | bool isValid() const | ||
| 376 | { | ||
| 377 | return !bFinished; | ||
| 378 | } | ||
| 379 | |||
| 380 | operator bool() const | ||
| 381 | { | ||
| 382 | return !bFinished; | ||
| 383 | } | ||
| 384 | } iterator; | ||
| 385 | |||
| 386 | /** | ||
| 387 | * Iteration structure for iterating through the set (const). | ||
| 388 | */ | ||
| 389 | typedef struct const_iterator | ||
| 390 | { | ||
| 391 | friend class Set<key, sizecalc, keyalloc, challoc>; | ||
| 392 | private: | ||
| 393 | const_iterator( const Set<key, sizecalc, keyalloc, challoc> &hsh ) : | ||
| 394 | hsh( hsh ), | ||
| 395 | nPos( 0 ), | ||
| 396 | bFinished( false ) | ||
| 397 | { | ||
| 398 | nPos = hsh.getFirstPos( bFinished ); | ||
| 399 | } | ||
| 400 | |||
| 401 | const_iterator( const Set<key, sizecalc, keyalloc, challoc> &hsh, bool bDone ) : | ||
| 402 | hsh( hsh ), | ||
| 403 | nPos( 0 ), | ||
| 404 | bFinished( bDone ) | ||
| 405 | { | ||
| 406 | } | ||
| 407 | |||
| 408 | const Set<key, sizecalc, keyalloc, challoc> &hsh; | ||
| 409 | uint32_t nPos; | ||
| 410 | bool bFinished; | ||
| 411 | |||
| 412 | public: | ||
| 413 | /** | ||
| 414 | * Iterator incrementation operator. Move the iterator forward. | ||
| 415 | */ | ||
| 416 | const_iterator operator++( int ) | ||
| 417 | { | ||
| 418 | if( bFinished == false ) | ||
| 419 | nPos = hsh.getNextPos( nPos, bFinished ); | ||
| 420 | |||
| 421 | return *this; | ||
| 422 | } | ||
| 423 | |||
| 424 | /** | ||
| 425 | * Iterator incrementation operator. Move the iterator forward. | ||
| 426 | */ | ||
| 427 | const_iterator operator++() | ||
| 428 | { | ||
| 429 | if( bFinished == false ) | ||
| 430 | nPos = hsh.getNextPos( nPos, bFinished ); | ||
| 431 | |||
| 432 | return *this; | ||
| 433 | } | ||
| 434 | |||
| 435 | /** | ||
| 436 | * Iterator equality comparison operator. Iterators the same? | ||
| 437 | */ | ||
| 438 | bool operator==( const const_iterator &oth ) const | ||
| 439 | { | ||
| 440 | if( bFinished != oth.bFinished ) | ||
| 441 | return false; | ||
| 442 | if( bFinished == true ) | ||
| 443 | { | ||
| 444 | return true; | ||
| 445 | } | ||
| 446 | else | ||
| 447 | { | ||
| 448 | if( oth.nPos == nPos ) | ||
| 449 | return true; | ||
| 450 | return false; | ||
| 451 | } | ||
| 452 | } | ||
| 453 | |||
| 454 | /** | ||
| 455 | * Iterator not equality comparison operator. Not the same? | ||
| 456 | */ | ||
| 457 | bool operator!=( const const_iterator &oth ) const | ||
| 458 | { | ||
| 459 | return !(*this == oth ); | ||
| 460 | } | ||
| 461 | |||
| 462 | /** | ||
| 463 | * Iterator assignment operator. | ||
| 464 | */ | ||
| 465 | const_iterator operator=( const const_iterator &oth ) | ||
| 466 | { | ||
| 467 | if( &hsh != &oth.hsh ) | ||
| 468 | throw SetException( | ||
| 469 | "Cannot mix iterators from different hash objects."); | ||
| 470 | nPos = oth.nPos; | ||
| 471 | bFinished = oth.bFinished; | ||
| 472 | } | ||
| 473 | |||
| 474 | /** | ||
| 475 | * Iterator dereference operator... err.. get the value | ||
| 476 | *@returns (value_type &) The value behind this iterator. | ||
| 477 | */ | ||
| 478 | const key &operator *() const | ||
| 479 | { | ||
| 480 | return hsh.getKeyAtPos( nPos ); | ||
| 481 | } | ||
| 482 | |||
| 483 | bool isValid() const | ||
| 484 | { | ||
| 485 | return !bFinished; | ||
| 486 | } | ||
| 487 | |||
| 488 | operator bool() const | ||
| 489 | { | ||
| 490 | return !bFinished; | ||
| 491 | } | ||
| 492 | } const_iterator; | ||
| 493 | |||
| 494 | /** | ||
| 495 | * Get an iterator pointing to the first item in the hash table. | ||
| 496 | *@returns (iterator) An iterator pointing to the first item in the | ||
| 497 | * hash table. | ||
| 498 | */ | ||
| 499 | iterator begin() | ||
| 500 | { | ||
| 501 | return iterator( *this ); | ||
| 502 | } | ||
| 503 | |||
| 504 | const_iterator begin() const | ||
| 505 | { | ||
| 506 | return const_iterator( *this ); | ||
| 507 | } | ||
| 508 | |||
| 509 | /** | ||
| 510 | * Get an iterator pointing to a point just past the last item in the | ||
| 511 | * hash table. | ||
| 512 | *@returns (iterator) An iterator pointing to a point just past the | ||
| 513 | * last item in the hash table. | ||
| 514 | */ | ||
| 515 | iterator end() | ||
| 516 | { | ||
| 517 | return iterator( *this, true ); | ||
| 518 | } | ||
| 519 | |||
| 520 | const_iterator end() const | ||
| 521 | { | ||
| 522 | return const_iterator( *this, true ); | ||
| 523 | } | ||
| 524 | |||
| 525 | /** | ||
| 526 | * Get a list of all the keys in the hash table. | ||
| 527 | *@returns (std::list<key_type>) The list of keys in the hash table. | ||
| 528 | */ | ||
| 529 | Bu::List<key> getKeys() const | ||
| 530 | { | ||
| 531 | Bu::List<key> lKeys; | ||
| 532 | |||
| 533 | for( uint32_t j = 0; j < nCapacity; j++ ) | ||
| 534 | { | ||
| 535 | if( isFilled( j ) ) | ||
| 536 | { | ||
| 537 | if( !isDeleted( j ) ) | ||
| 538 | { | ||
| 539 | lKeys.append( aKeys[j] ); | ||
| 540 | } | ||
| 541 | } | ||
| 542 | } | ||
| 543 | |||
| 544 | return lKeys; | ||
| 545 | } | ||
| 546 | |||
| 547 | protected: | ||
| 548 | virtual void onInsert() {} | ||
| 549 | virtual void onUpdate() {} | ||
| 550 | virtual void onDelete() {} | ||
| 551 | virtual void onReHash() {} | ||
| 552 | |||
| 553 | virtual void clearBits() | ||
| 554 | { | ||
| 555 | for( uint32_t j = 0; j < nKeysSize; j++ ) | ||
| 556 | { | ||
| 557 | bFilled[j] = bDeleted[j] = 0; | ||
| 558 | } | ||
| 559 | } | ||
| 560 | |||
| 561 | virtual void fill( uint32_t loc, key &k, uint32_t hash ) | ||
| 562 | { | ||
| 563 | bFilled[loc/32] |= (1<<(loc%32)); | ||
| 564 | ka.construct( &aKeys[loc], k ); | ||
| 565 | aHashCodes[loc] = hash; | ||
| 566 | nFilled++; | ||
| 567 | //printf("Filled: %d, Deleted: %d, Capacity: %d\n", | ||
| 568 | // nFilled, nDeleted, nCapacity ); | ||
| 569 | } | ||
| 570 | |||
| 571 | virtual void _erase( uint32_t loc ) | ||
| 572 | { | ||
| 573 | bDeleted[loc/32] |= (1<<(loc%32)); | ||
| 574 | ka.destroy( &aKeys[loc] ); | ||
| 575 | nDeleted++; | ||
| 576 | //printf("Filled: %d, Deleted: %d, Capacity: %d\n", | ||
| 577 | // nFilled, nDeleted, nCapacity ); | ||
| 578 | } | ||
| 579 | |||
| 580 | virtual key &getKeyAtPos( uint32_t nPos ) | ||
| 581 | { | ||
| 582 | return aKeys[nPos]; | ||
| 583 | } | ||
| 584 | |||
| 585 | virtual const key &getKeyAtPos( uint32_t nPos ) const | ||
| 586 | { | ||
| 587 | return aKeys[nPos]; | ||
| 588 | } | ||
| 589 | |||
| 590 | virtual uint32_t getFirstPos( bool &bFinished ) const | ||
| 591 | { | ||
| 592 | for( uint32_t j = 0; j < nCapacity; j++ ) | ||
| 593 | { | ||
| 594 | if( isFilled( j ) ) | ||
| 595 | if( !isDeleted( j ) ) | ||
| 596 | return j; | ||
| 597 | } | ||
| 598 | |||
| 599 | bFinished = true; | ||
| 600 | return 0; | ||
| 601 | } | ||
| 602 | |||
| 603 | virtual uint32_t getNextPos( uint32_t nPos, bool &bFinished ) const | ||
| 604 | { | ||
| 605 | for( uint32_t j = nPos+1; j < nCapacity; j++ ) | ||
| 606 | { | ||
| 607 | if( isFilled( j ) ) | ||
| 608 | if( !isDeleted( j ) ) | ||
| 609 | return j; | ||
| 610 | } | ||
| 611 | |||
| 612 | bFinished = true; | ||
| 613 | return 0; | ||
| 614 | } | ||
| 615 | |||
| 616 | uint32_t probe( uint32_t hash, key k, bool &bFill, bool rehash=true ) | ||
| 617 | { | ||
| 618 | uint32_t nCur = hash%nCapacity; | ||
| 619 | |||
| 620 | // First we scan to see if the key is already there, abort if we | ||
| 621 | // run out of probing room, or we find a non-filled entry | ||
| 622 | int8_t j; | ||
| 623 | for( j = 0; | ||
| 624 | isFilled( nCur ) && j < 32; | ||
| 625 | nCur = (nCur + (1<<j))%nCapacity, j++ | ||
| 626 | ) | ||
| 627 | { | ||
| 628 | // Is this the same hash code we were looking for? | ||
| 629 | if( hash == aHashCodes[nCur] ) | ||
| 630 | { | ||
| 631 | // Skip over deleted entries. Deleted entries are also filled, | ||
| 632 | // so we only have to do this check here. | ||
| 633 | if( isDeleted( nCur ) ) | ||
| 634 | continue; | ||
| 635 | |||
| 636 | // Is it really the same key? (for safety) | ||
| 637 | if( __cmpHashKeys( aKeys[nCur], k ) == true ) | ||
| 638 | { | ||
| 639 | bFill = true; | ||
| 640 | return nCur; | ||
| 641 | } | ||
| 642 | } | ||
| 643 | } | ||
| 644 | |||
| 645 | // This is our insurance, if the table is full, then go ahead and | ||
| 646 | // rehash, then try again. | ||
| 647 | if( (isFilled( nCur ) || j == 32) && rehash == true ) | ||
| 648 | { | ||
| 649 | reHash( szCalc(getCapacity(), getFill(), getDeleted()) ); | ||
| 650 | |||
| 651 | // This is potentially dangerous, and could cause an infinite loop. | ||
| 652 | // Be careful writing probe, eh? | ||
| 653 | return probe( hash, k, bFill ); | ||
| 654 | } | ||
| 655 | |||
| 656 | bFill = false; | ||
| 657 | return nCur; | ||
| 658 | } | ||
| 659 | |||
| 660 | uint32_t probe( uint32_t hash, key k, bool &bFill, bool rehash=true ) const | ||
| 661 | { | ||
| 662 | uint32_t nCur = hash%nCapacity; | ||
| 663 | |||
| 664 | // First we scan to see if the key is already there, abort if we | ||
| 665 | // run out of probing room, or we find a non-filled entry | ||
| 666 | for( int8_t j = 0; | ||
| 667 | isFilled( nCur ) && j < 32; | ||
| 668 | nCur = (nCur + (1<<j))%nCapacity, j++ | ||
| 669 | ) | ||
| 670 | { | ||
| 671 | // Is this the same hash code we were looking for? | ||
| 672 | if( hash == aHashCodes[nCur] ) | ||
| 673 | { | ||
| 674 | // Skip over deleted entries. Deleted entries are also filled, | ||
| 675 | // so we only have to do this check here. | ||
| 676 | if( isDeleted( nCur ) ) | ||
| 677 | continue; | ||
| 678 | |||
| 679 | // Is it really the same key? (for safety) | ||
| 680 | if( __cmpHashKeys( aKeys[nCur], k ) == true ) | ||
| 681 | { | ||
| 682 | bFill = true; | ||
| 683 | return nCur; | ||
| 684 | } | ||
| 685 | } | ||
| 686 | } | ||
| 687 | |||
| 688 | bFill = false; | ||
| 689 | return nCur; | ||
| 690 | } | ||
| 691 | |||
| 692 | void reHash( uint32_t nNewSize ) | ||
| 693 | { | ||
| 694 | //printf("---REHASH---"); | ||
| 695 | //printf("Filled: %d, Deleted: %d, Capacity: %d\n", | ||
| 696 | // nFilled, nDeleted, nCapacity ); | ||
| 697 | |||
| 698 | // Save all the old data | ||
| 699 | uint32_t nOldCapacity = nCapacity; | ||
| 700 | uint32_t *bOldFilled = bFilled; | ||
| 701 | uint32_t *aOldHashCodes = aHashCodes; | ||
| 702 | uint32_t nOldKeysSize = nKeysSize; | ||
| 703 | uint32_t *bOldDeleted = bDeleted; | ||
| 704 | key *aOldKeys = aKeys; | ||
| 705 | |||
| 706 | // Calculate new sizes | ||
| 707 | nCapacity = nNewSize; | ||
| 708 | nKeysSize = bitsToBytes( nCapacity ); | ||
| 709 | |||
| 710 | // Allocate new memory + prep | ||
| 711 | bFilled = ca.allocate( nKeysSize ); | ||
| 712 | bDeleted = ca.allocate( nKeysSize ); | ||
| 713 | clearBits(); | ||
| 714 | |||
| 715 | aHashCodes = ca.allocate( nCapacity ); | ||
| 716 | aKeys = ka.allocate( nCapacity ); | ||
| 717 | |||
| 718 | nDeleted = nFilled = 0; | ||
| 719 | |||
| 720 | // Re-insert all of the old data (except deleted items) | ||
| 721 | for( uint32_t j = 0; j < nOldCapacity; j++ ) | ||
| 722 | { | ||
| 723 | if( (bOldFilled[j/32]&(1<<(j%32)))!=0 && | ||
| 724 | (bOldDeleted[j/32]&(1<<(j%32)))==0 ) | ||
| 725 | { | ||
| 726 | insert( aOldKeys[j] ); | ||
| 727 | } | ||
| 728 | } | ||
| 729 | |||
| 730 | // Delete all of the old data | ||
| 731 | for( uint32_t j = 0; j < nOldCapacity; j++ ) | ||
| 732 | { | ||
| 733 | if( (bOldFilled[j/32]&(1<<(j%32)))!=0 ) | ||
| 734 | { | ||
| 735 | ka.destroy( &aOldKeys[j] ); | ||
| 736 | } | ||
| 737 | } | ||
| 738 | ka.deallocate( aOldKeys, nOldCapacity ); | ||
| 739 | ca.deallocate( bOldFilled, nOldKeysSize ); | ||
| 740 | ca.deallocate( bOldDeleted, nOldKeysSize ); | ||
| 741 | ca.deallocate( aOldHashCodes, nOldCapacity ); | ||
| 742 | } | ||
| 743 | |||
| 744 | virtual bool isFilled( uint32_t loc ) const | ||
| 745 | { | ||
| 746 | return (bFilled[loc/32]&(1<<(loc%32)))!=0; | ||
| 747 | } | ||
| 748 | |||
| 749 | virtual bool isDeleted( uint32_t loc ) const | ||
| 750 | { | ||
| 751 | return (bDeleted[loc/32]&(1<<(loc%32)))!=0; | ||
| 752 | } | ||
| 753 | |||
| 754 | protected: | ||
| 755 | uint32_t nCapacity; | ||
| 756 | uint32_t nFilled; | ||
| 757 | uint32_t nDeleted; | ||
| 758 | uint32_t *bFilled; | ||
| 759 | uint32_t *bDeleted; | ||
| 760 | uint32_t nKeysSize; | ||
| 761 | key *aKeys; | ||
| 762 | uint32_t *aHashCodes; | ||
| 763 | keyalloc ka; | ||
| 764 | challoc ca; | ||
| 765 | sizecalc szCalc; | ||
| 766 | }; | 41 | }; | 
| 767 | |||
| 768 | template<typename key, typename b, typename c, typename d> | ||
| 769 | Archive &operator<<( Archive &ar, const Set<key, b, c, d> &h ) | ||
| 770 | { | ||
| 771 | ar << h.getSize(); | ||
| 772 | for( typename Set<key, b, c, d>::const_iterator i = h.begin(); i != h.end(); i++ ) | ||
| 773 | { | ||
| 774 | ar << (*i); | ||
| 775 | } | ||
| 776 | |||
| 777 | return ar; | ||
| 778 | } | ||
| 779 | |||
| 780 | template<typename key, typename b, typename c, typename d> | ||
| 781 | Archive &operator>>( Archive &ar, Set<key, b, c, d> &h ) | ||
| 782 | { | ||
| 783 | h.clear(); | ||
| 784 | long nSize; | ||
| 785 | ar >> nSize; | ||
| 786 | |||
| 787 | for( long j = 0; j < nSize; j++ ) | ||
| 788 | { | ||
| 789 | key v; | ||
| 790 | ar >> v; | ||
| 791 | h.insert( v ); | ||
| 792 | } | ||
| 793 | |||
| 794 | return ar; | ||
| 795 | } | ||
| 796 | } | 42 | } | 
| 797 | 43 | ||
| 798 | #endif | 44 | #endif | 
| diff --git a/src/sharedcore.h b/src/sharedcore.h index 5a44df9..1887ca2 100644 --- a/src/sharedcore.h +++ b/src/sharedcore.h | |||
| @@ -15,10 +15,57 @@ | |||
| 15 | 15 | ||
| 16 | namespace Bu | 16 | namespace Bu | 
| 17 | { | 17 | { | 
| 18 | template<typename Core> | 18 | /** | 
| 19 | * A mechanism for creating classes that perform lazy copies. The concept | ||
| 20 | * behind this is that instead of copying a large object when it is assigned | ||
| 21 | * or passed into a copy constructor we simply copy a pointer internally. | ||
| 22 | * The assumption is that many times when an object is passed by value we | ||
| 23 | * don't really want to keep the object around, we want the recipient to | ||
| 24 | * take ownership without allocating a new object. This allows that to | ||
| 25 | * happen. | ||
| 26 | * | ||
| 27 | * When used properly this makes object copying essentially free (O(1), | ||
| 28 | * that is) and performs the actual copy when a user tries to modify the | ||
| 29 | * object. | ||
| 30 | * | ||
| 31 | * For example, lets look at something like the getKeys function in | ||
| 32 | * Bu::Hash. When this function is called it creates a Bu::List of | ||
| 33 | * appropriate type, fills it with keys, and returns it. This is a good | ||
| 34 | * way for this function to behave, there may be additional issues if the | ||
| 35 | * List object were allocated with new and not on the stack. However, | ||
| 36 | * returning the List at the end of the function could potentially take | ||
| 37 | * a very long time depending on the size of the list and the type of the | ||
| 38 | * key. In this case the getKeys function doesn't want ownership of the | ||
| 39 | * List object, and when it returns it, it's local copy will be destroyed. | ||
| 40 | * | ||
| 41 | * However, List inherits from SharedCore, which means that when it is | ||
| 42 | * returned all we do is copy a pointer to the "core" of the list, which | ||
| 43 | * is a very fast operatorion. For a brief moment, before anyone can do | ||
| 44 | * anything else, there are two objects referencing the core of that single | ||
| 45 | * list. However, the getKeys() function will destroy it's local copy | ||
| 46 | * before the calling function can use it's new copy. That means that by | ||
| 47 | * the time the calling function can use it's new List of keys it is the | ||
| 48 | * only one with a reference to the core, and no copy will need to happen. | ||
| 49 | * | ||
| 50 | * Using SharedCore on your own classes is fairly straight forward. There | ||
| 51 | * are only a couple of steps. First, break the class into two classes. | ||
| 52 | * Move every variable from the original class (generally everything that's | ||
| 53 | * private) into the new class. Then make the original class inherit from | ||
| 54 | * SharedCore. The SharedCore template takes 2 parameters, first is the | ||
| 55 | * class it's inheriting from, second is the new core class. Now, in your | ||
| 56 | * original class you will have one class variable, a pointer named core. | ||
| 57 | * All of your original variables will be accessable through core. The next | ||
| 58 | * step is to access everything you used to through core, and to find | ||
| 59 | * every function that may change data in the core. At the top of every | ||
| 60 | * function that may change data you want to call _hardCopy(). | ||
| 61 | * | ||
| 62 | * That's more or less it. A more detailed guide will be written soon. | ||
| 63 | * @todo Write a guide for this. | ||
| 64 | */ | ||
| 65 | template<typename Shell, typename Core> | ||
| 19 | class SharedCore | 66 | class SharedCore | 
| 20 | { | 67 | { | 
| 21 | typedef class SharedCore<Core> _SharedType; | 68 | typedef class SharedCore<Shell, Core> _SharedType; | 
| 22 | public: | 69 | public: | 
| 23 | SharedCore() : | 70 | SharedCore() : | 
| 24 | core( NULL ), | 71 | core( NULL ), | 
| @@ -54,6 +101,18 @@ namespace Bu | |||
| 54 | return *iRefCount; | 101 | return *iRefCount; | 
| 55 | } | 102 | } | 
| 56 | 103 | ||
| 104 | Shell clone() const | ||
| 105 | { | ||
| 106 | Shell s( dynamic_cast<const Shell &>(*this) ); | ||
| 107 | s._hardCopy(); | ||
| 108 | return s; | ||
| 109 | } | ||
| 110 | |||
| 111 | bool isCoreShared( const Shell &rOther ) const | ||
| 112 | { | ||
| 113 | return rOther.core == core; | ||
| 114 | } | ||
| 115 | |||
| 57 | protected: | 116 | protected: | 
| 58 | Core *core; | 117 | Core *core; | 
| 59 | void _hardCopy() | 118 | void _hardCopy() | 
| @@ -68,6 +127,20 @@ namespace Bu | |||
| 68 | iRefCount = new int( 1 ); | 127 | iRefCount = new int( 1 ); | 
| 69 | } | 128 | } | 
| 70 | 129 | ||
| 130 | /** | ||
| 131 | * Reset core acts like a hard copy, except instead of providing a | ||
| 132 | * standalone copy of the shared core, it provides a brand new core. | ||
| 133 | * | ||
| 134 | * Very useful in functions used to reset the state of an object. | ||
| 135 | */ | ||
| 136 | void _resetCore() | ||
| 137 | { | ||
| 138 | if( core ) | ||
| 139 | _deref(); | ||
| 140 | core = _allocateCore(); | ||
| 141 | iRefCount = new int( 1 ); | ||
| 142 | } | ||
| 143 | |||
| 71 | virtual Core *_allocateCore() | 144 | virtual Core *_allocateCore() | 
| 72 | { | 145 | { | 
| 73 | return new Core(); | 146 | return new Core(); | 
| diff --git a/src/serversocket.cpp b/src/tcpserversocket.cpp index 87d0035..7d7f6e4 100644 --- a/src/serversocket.cpp +++ b/src/tcpserversocket.cpp | |||
| @@ -21,13 +21,13 @@ | |||
| 21 | #include <sys/types.h> | 21 | #include <sys/types.h> | 
| 22 | //#include <termios.h> | 22 | //#include <termios.h> | 
| 23 | #include <fcntl.h> | 23 | #include <fcntl.h> | 
| 24 | #include "bu/serversocket.h" | 24 | #include "bu/tcpserversocket.h" | 
| 25 | 25 | ||
| 26 | #include "bu/config.h" | 26 | #include "bu/config.h" | 
| 27 | 27 | ||
| 28 | namespace Bu { subExceptionDef( ServerSocketException ) } | 28 | namespace Bu { subExceptionDef( TcpServerSocketException ) } | 
| 29 | 29 | ||
| 30 | Bu::ServerSocket::ServerSocket( int nPort, int nPoolSize ) : | 30 | Bu::TcpServerSocket::TcpServerSocket( int nPort, int nPoolSize ) : | 
| 31 | nPort( nPort ) | 31 | nPort( nPort ) | 
| 32 | { | 32 | { | 
| 33 | #ifdef WIN32 | 33 | #ifdef WIN32 | 
| @@ -48,7 +48,7 @@ Bu::ServerSocket::ServerSocket( int nPort, int nPoolSize ) : | |||
| 48 | startServer( name, nPoolSize ); | 48 | startServer( name, nPoolSize ); | 
| 49 | } | 49 | } | 
| 50 | 50 | ||
| 51 | Bu::ServerSocket::ServerSocket(const FString &sAddr,int nPort, int nPoolSize) : | 51 | Bu::TcpServerSocket::TcpServerSocket(const FString &sAddr,int nPort, int nPoolSize) : | 
| 52 | nPort( nPort ) | 52 | nPort( nPort ) | 
| 53 | { | 53 | { | 
| 54 | #ifdef WIN32 | 54 | #ifdef WIN32 | 
| @@ -72,7 +72,7 @@ Bu::ServerSocket::ServerSocket(const FString &sAddr,int nPort, int nPoolSize) : | |||
| 72 | startServer( name, nPoolSize ); | 72 | startServer( name, nPoolSize ); | 
| 73 | } | 73 | } | 
| 74 | 74 | ||
| 75 | Bu::ServerSocket::ServerSocket( int nServer, bool bInit, int nPoolSize ) : | 75 | Bu::TcpServerSocket::TcpServerSocket( int nServer, bool bInit, int nPoolSize ) : | 
| 76 | nServer( nServer ), | 76 | nServer( nServer ), | 
| 77 | nPort( 0 ) | 77 | nPort( 0 ) | 
| 78 | { | 78 | { | 
| @@ -95,7 +95,7 @@ Bu::ServerSocket::ServerSocket( int nServer, bool bInit, int nPoolSize ) : | |||
| 95 | } | 95 | } | 
| 96 | } | 96 | } | 
| 97 | 97 | ||
| 98 | Bu::ServerSocket::ServerSocket( const ServerSocket &rSrc ) | 98 | Bu::TcpServerSocket::TcpServerSocket( const TcpServerSocket &rSrc ) | 
| 99 | { | 99 | { | 
| 100 | #ifdef WIN32 | 100 | #ifdef WIN32 | 
| 101 | Bu::Winsock2::getInstance(); | 101 | Bu::Winsock2::getInstance(); | 
| @@ -107,20 +107,20 @@ Bu::ServerSocket::ServerSocket( const ServerSocket &rSrc ) | |||
| 107 | FD_SET( nServer, &fdActive ); | 107 | FD_SET( nServer, &fdActive ); | 
| 108 | } | 108 | } | 
| 109 | 109 | ||
| 110 | Bu::ServerSocket::~ServerSocket() | 110 | Bu::TcpServerSocket::~TcpServerSocket() | 
| 111 | { | 111 | { | 
| 112 | if( nServer > -1 ) | 112 | if( nServer > -1 ) | 
| 113 | ::close( nServer ); | 113 | ::close( nServer ); | 
| 114 | } | 114 | } | 
| 115 | 115 | ||
| 116 | void Bu::ServerSocket::startServer( struct sockaddr_in &name, int nPoolSize ) | 116 | void Bu::TcpServerSocket::startServer( struct sockaddr_in &name, int nPoolSize ) | 
| 117 | { | 117 | { | 
| 118 | /* Create the socket. */ | 118 | /* Create the socket. */ | 
| 119 | nServer = bu_socket( PF_INET, SOCK_STREAM, 0 ); | 119 | nServer = bu_socket( PF_INET, SOCK_STREAM, 0 ); | 
| 120 | 120 | ||
| 121 | if( nServer < 0 ) | 121 | if( nServer < 0 ) | 
| 122 | { | 122 | { | 
| 123 | throw Bu::ServerSocketException("Couldn't create a listen socket."); | 123 | throw Bu::TcpServerSocketException("Couldn't create a listen socket."); | 
| 124 | } | 124 | } | 
| 125 | 125 | ||
| 126 | int opt = 1; | 126 | int opt = 1; | 
| @@ -135,16 +135,16 @@ void Bu::ServerSocket::startServer( struct sockaddr_in &name, int nPoolSize ) | |||
| 135 | initServer( name, nPoolSize ); | 135 | initServer( name, nPoolSize ); | 
| 136 | } | 136 | } | 
| 137 | 137 | ||
| 138 | void Bu::ServerSocket::initServer( struct sockaddr_in &name, int nPoolSize ) | 138 | void Bu::TcpServerSocket::initServer( struct sockaddr_in &name, int nPoolSize ) | 
| 139 | { | 139 | { | 
| 140 | if( bu_bind( nServer, (struct sockaddr *) &name, sizeof(name) ) < 0 ) | 140 | if( bu_bind( nServer, (struct sockaddr *) &name, sizeof(name) ) < 0 ) | 
| 141 | { | 141 | { | 
| 142 | throw Bu::ServerSocketException("Couldn't bind to the listen socket."); | 142 | throw Bu::TcpServerSocketException("Couldn't bind to the listen socket."); | 
| 143 | } | 143 | } | 
| 144 | 144 | ||
| 145 | if( bu_listen( nServer, nPoolSize ) < 0 ) | 145 | if( bu_listen( nServer, nPoolSize ) < 0 ) | 
| 146 | { | 146 | { | 
| 147 | throw Bu::ServerSocketException( | 147 | throw Bu::TcpServerSocketException( | 
| 148 | "Couldn't begin listening to the server socket." | 148 | "Couldn't begin listening to the server socket." | 
| 149 | ); | 149 | ); | 
| 150 | } | 150 | } | 
| @@ -154,12 +154,12 @@ void Bu::ServerSocket::initServer( struct sockaddr_in &name, int nPoolSize ) | |||
| 154 | FD_SET( nServer, &fdActive ); | 154 | FD_SET( nServer, &fdActive ); | 
| 155 | } | 155 | } | 
| 156 | 156 | ||
| 157 | int Bu::ServerSocket::getSocket() | 157 | int Bu::TcpServerSocket::getSocket() | 
| 158 | { | 158 | { | 
| 159 | return nServer; | 159 | return nServer; | 
| 160 | } | 160 | } | 
| 161 | 161 | ||
| 162 | int Bu::ServerSocket::accept( int nTimeoutSec, int nTimeoutUSec ) | 162 | int Bu::TcpServerSocket::accept( int nTimeoutSec, int nTimeoutUSec ) | 
| 163 | { | 163 | { | 
| 164 | fd_set fdRead = fdActive; | 164 | fd_set fdRead = fdActive; | 
| 165 | 165 | ||
| @@ -171,7 +171,7 @@ int Bu::ServerSocket::accept( int nTimeoutSec, int nTimeoutUSec ) | |||
| 171 | if( TEMP_FAILURE_RETRY( | 171 | if( TEMP_FAILURE_RETRY( | 
| 172 | bu_select( nServer+1, &fdRead, NULL, NULL, &xT )) < 0 ) | 172 | bu_select( nServer+1, &fdRead, NULL, NULL, &xT )) < 0 ) | 
| 173 | { | 173 | { | 
| 174 | throw Bu::ServerSocketException( | 174 | throw Bu::TcpServerSocketException( | 
| 175 | "Error scanning for new connections: %s", strerror( errno ) | 175 | "Error scanning for new connections: %s", strerror( errno ) | 
| 176 | ); | 176 | ); | 
| 177 | } | 177 | } | 
| @@ -200,7 +200,7 @@ int Bu::ServerSocket::accept( int nTimeoutSec, int nTimeoutUSec ) | |||
| 200 | #endif /* WIN32 */ | 200 | #endif /* WIN32 */ | 
| 201 | if( nClient < 0 ) | 201 | if( nClient < 0 ) | 
| 202 | { | 202 | { | 
| 203 | throw Bu::ServerSocketException( | 203 | throw Bu::TcpServerSocketException( | 
| 204 | "Error accepting a new connection: %s", strerror( errno ) | 204 | "Error accepting a new connection: %s", strerror( errno ) | 
| 205 | ); | 205 | ); | 
| 206 | } | 206 | } | 
| @@ -219,7 +219,7 @@ int Bu::ServerSocket::accept( int nTimeoutSec, int nTimeoutUSec ) | |||
| 219 | flags |= O_NONBLOCK; | 219 | flags |= O_NONBLOCK; | 
| 220 | if( fcntl( nClient, F_SETFL, flags ) < 0) | 220 | if( fcntl( nClient, F_SETFL, flags ) < 0) | 
| 221 | { | 221 | { | 
| 222 | throw Bu::ServerSocketException( | 222 | throw Bu::TcpServerSocketException( | 
| 223 | "Error setting option on client socket: %s", | 223 | "Error setting option on client socket: %s", | 
| 224 | strerror( errno ) | 224 | strerror( errno ) | 
| 225 | ); | 225 | ); | 
| @@ -242,7 +242,7 @@ int Bu::ServerSocket::accept( int nTimeoutSec, int nTimeoutUSec ) | |||
| 242 | return -1; | 242 | return -1; | 
| 243 | } | 243 | } | 
| 244 | 244 | ||
| 245 | int Bu::ServerSocket::getPort() | 245 | int Bu::TcpServerSocket::getPort() | 
| 246 | { | 246 | { | 
| 247 | return nPort; | 247 | return nPort; | 
| 248 | } | 248 | } | 
| diff --git a/src/serversocket.h b/src/tcpserversocket.h index ee357a4..b1d7e02 100644 --- a/src/serversocket.h +++ b/src/tcpserversocket.h | |||
| @@ -5,8 +5,8 @@ | |||
| 5 | * terms of the license contained in the file LICENSE. | 5 | * terms of the license contained in the file LICENSE. | 
| 6 | */ | 6 | */ | 
| 7 | 7 | ||
| 8 | #ifndef BU_SERVER_SOCKET_H | 8 | #ifndef BU_TCP_SERVER_SOCKET_H | 
| 9 | #define BU_SERVER_SOCKET_H | 9 | #define BU_TCP_SERVER_SOCKET_H | 
| 10 | 10 | ||
| 11 | #include <stdint.h> | 11 | #include <stdint.h> | 
| 12 | #include "bu/fstring.h" | 12 | #include "bu/fstring.h" | 
| @@ -20,7 +20,7 @@ | |||
| 20 | 20 | ||
| 21 | namespace Bu | 21 | namespace Bu | 
| 22 | { | 22 | { | 
| 23 | subExceptionDecl( ServerSocketException ); | 23 | subExceptionDecl( TcpServerSocketException ); | 
| 24 | 24 | ||
| 25 | /** | 25 | /** | 
| 26 | * A single tcp/ip server socket. When created the server socket will bind | 26 | * A single tcp/ip server socket. When created the server socket will bind | 
| @@ -34,14 +34,14 @@ namespace Bu | |||
| 34 | * | 34 | * | 
| 35 | *@ingroup Serving | 35 | *@ingroup Serving | 
| 36 | */ | 36 | */ | 
| 37 | class ServerSocket | 37 | class TcpServerSocket | 
| 38 | { | 38 | { | 
| 39 | public: | 39 | public: | 
| 40 | ServerSocket( int nPort, int nPoolSize=40 ); | 40 | TcpServerSocket( int nPort, int nPoolSize=40 ); | 
| 41 | ServerSocket( const FString &sAddr, int nPort, int nPoolSize=40 ); | 41 | TcpServerSocket( const FString &sAddr, int nPort, int nPoolSize=40 ); | 
| 42 | ServerSocket( int nSocket, bool bInit, int nPoolSize=40 ); | 42 | TcpServerSocket( int nSocket, bool bInit, int nPoolSize=40 ); | 
| 43 | ServerSocket( const ServerSocket &rSrc ); | 43 | TcpServerSocket( const TcpServerSocket &rSrc ); | 
| 44 | virtual ~ServerSocket(); | 44 | virtual ~TcpServerSocket(); | 
| 45 | 45 | ||
| 46 | int accept( int nTimeoutSec=0, int nTimeoutUSec=0 ); | 46 | int accept( int nTimeoutSec=0, int nTimeoutUSec=0 ); | 
| 47 | int getSocket(); | 47 | int getSocket(); | 
| diff --git a/src/socket.cpp b/src/tcpsocket.cpp index baf3be3..bbd9cf5 100644 --- a/src/socket.cpp +++ b/src/tcpsocket.cpp | |||
| @@ -14,7 +14,7 @@ | |||
| 14 | #include <sys/time.h> | 14 | #include <sys/time.h> | 
| 15 | #include <errno.h> | 15 | #include <errno.h> | 
| 16 | #include <fcntl.h> | 16 | #include <fcntl.h> | 
| 17 | #include "bu/socket.h" | 17 | #include "bu/tcpsocket.h" | 
| 18 | 18 | ||
| 19 | #include "bu/config.h" | 19 | #include "bu/config.h" | 
| 20 | 20 | ||
| @@ -29,10 +29,10 @@ | |||
| 29 | 29 | ||
| 30 | #define RBS (1024*2) | 30 | #define RBS (1024*2) | 
| 31 | 31 | ||
| 32 | namespace Bu { subExceptionDef( SocketException ) } | 32 | namespace Bu { subExceptionDef( TcpSocketException ) } | 
| 33 | 33 | ||
| 34 | Bu::Socket::Socket( int nSocket ) : | 34 | Bu::TcpSocket::TcpSocket( int nTcpSocket ) : | 
| 35 | nSocket( nSocket ), | 35 | nTcpSocket( nTcpSocket ), | 
| 36 | bActive( true ), | 36 | bActive( true ), | 
| 37 | bBlocking( true ) | 37 | bBlocking( true ) | 
| 38 | { | 38 | { | 
| @@ -42,8 +42,9 @@ Bu::Socket::Socket( int nSocket ) : | |||
| 42 | setAddress(); | 42 | setAddress(); | 
| 43 | } | 43 | } | 
| 44 | 44 | ||
| 45 | Bu::Socket::Socket( const Bu::FString &sAddr, int nPort, int nTimeout ) : | 45 | Bu::TcpSocket::TcpSocket( const Bu::FString &sAddr, int nPort, int nTimeout, | 
| 46 | nSocket( 0 ), | 46 | bool bBlocking ) : | 
| 47 | nTcpSocket( 0 ), | ||
| 47 | bActive( false ), | 48 | bActive( false ), | 
| 48 | bBlocking( true ) | 49 | bBlocking( true ) | 
| 49 | { | 50 | { | 
| @@ -52,9 +53,9 @@ Bu::Socket::Socket( const Bu::FString &sAddr, int nPort, int nTimeout ) : | |||
| 52 | #endif | 53 | #endif | 
| 53 | 54 | ||
| 54 | /* Create the socket. */ | 55 | /* Create the socket. */ | 
| 55 | nSocket = bu_socket( PF_INET, SOCK_STREAM, 0 ); | 56 | nTcpSocket = bu_socket( PF_INET, SOCK_STREAM, 0 ); | 
| 56 | 57 | ||
| 57 | if( nSocket < 0 ) | 58 | if( nTcpSocket < 0 ) | 
| 58 | { | 59 | { | 
| 59 | throw ExceptionBase("Couldn't create socket.\n"); | 60 | throw ExceptionBase("Couldn't create socket.\n"); | 
| 60 | } | 61 | } | 
| @@ -78,12 +79,12 @@ Bu::Socket::Socket( const Bu::FString &sAddr, int nPort, int nTimeout ) : | |||
| 78 | sAddr.getStr(), ibuf, &aiHints, &pAddr )) != 0 ) | 79 | sAddr.getStr(), ibuf, &aiHints, &pAddr )) != 0 ) | 
| 79 | { | 80 | { | 
| 80 | close(); | 81 | close(); | 
| 81 | throw Bu::SocketException("Couldn't resolve hostname %s (%s).\n", | 82 | throw Bu::TcpSocketException("Couldn't resolve hostname %s (%s).\n", | 
| 82 | sAddr.getStr(), bu_gai_strerror(ret)); | 83 | sAddr.getStr(), bu_gai_strerror(ret)); | 
| 83 | } | 84 | } | 
| 84 | 85 | ||
| 85 | bu_connect( | 86 | bu_connect( | 
| 86 | nSocket, | 87 | nTcpSocket, | 
| 87 | pAddr->ai_addr, | 88 | pAddr->ai_addr, | 
| 88 | pAddr->ai_addrlen | 89 | pAddr->ai_addrlen | 
| 89 | ); | 90 | ); | 
| @@ -101,17 +102,17 @@ Bu::Socket::Socket( const Bu::FString &sAddr, int nPort, int nTimeout ) : | |||
| 101 | int retval; | 102 | int retval; | 
| 102 | 103 | ||
| 103 | FD_ZERO(&rfds); | 104 | FD_ZERO(&rfds); | 
| 104 | FD_SET(nSocket, &rfds); | 105 | FD_SET(nTcpSocket, &rfds); | 
| 105 | FD_ZERO(&wfds); | 106 | FD_ZERO(&wfds); | 
| 106 | FD_SET(nSocket, &wfds); | 107 | FD_SET(nTcpSocket, &wfds); | 
| 107 | FD_ZERO(&efds); | 108 | FD_ZERO(&efds); | 
| 108 | FD_SET(nSocket, &efds); | 109 | FD_SET(nTcpSocket, &efds); | 
| 109 | 110 | ||
| 110 | struct timeval tv; | 111 | struct timeval tv; | 
| 111 | tv.tv_sec = nTimeout; | 112 | tv.tv_sec = nTimeout; | 
| 112 | tv.tv_usec = 0; | 113 | tv.tv_usec = 0; | 
| 113 | 114 | ||
| 114 | retval = bu_select( nSocket+1, &rfds, &wfds, &efds, &tv ); | 115 | retval = bu_select( nTcpSocket+1, &rfds, &wfds, &efds, &tv ); | 
| 115 | 116 | ||
| 116 | if( retval == 0 ) | 117 | if( retval == 0 ) | 
| 117 | { | 118 | { | 
| @@ -120,51 +121,54 @@ Bu::Socket::Socket( const Bu::FString &sAddr, int nPort, int nTimeout ) : | |||
| 120 | } | 121 | } | 
| 121 | read( NULL, 0 ); // See if we can get any errors out of the way early. | 122 | read( NULL, 0 ); // See if we can get any errors out of the way early. | 
| 122 | } | 123 | } | 
| 124 | |||
| 125 | if( bBlocking ) | ||
| 126 | setBlocking( bBlocking ); | ||
| 123 | } | 127 | } | 
| 124 | 128 | ||
| 125 | Bu::Socket::~Socket() | 129 | Bu::TcpSocket::~TcpSocket() | 
| 126 | { | 130 | { | 
| 127 | close(); | 131 | close(); | 
| 128 | } | 132 | } | 
| 129 | 133 | ||
| 130 | void Bu::Socket::close() | 134 | void Bu::TcpSocket::close() | 
| 131 | { | 135 | { | 
| 132 | if( bActive ) | 136 | if( bActive ) | 
| 133 | { | 137 | { | 
| 134 | #ifndef WIN32 | 138 | #ifndef WIN32 | 
| 135 | fsync( nSocket ); | 139 | fsync( nTcpSocket ); | 
| 136 | #endif | 140 | #endif | 
| 137 | #ifdef WIN32 | 141 | #ifdef WIN32 | 
| 138 | #ifndef SHUT_RDWR | 142 | #ifndef SHUT_RDWR | 
| 139 | #define SHUT_RDWR (SD_BOTH) | 143 | #define SHUT_RDWR (SD_BOTH) | 
| 140 | #endif | 144 | #endif | 
| 141 | #endif | 145 | #endif | 
| 142 | bu_shutdown( nSocket, SHUT_RDWR ); | 146 | bu_shutdown( nTcpSocket, SHUT_RDWR ); | 
| 143 | ::close( nSocket ); | 147 | ::close( nTcpSocket ); | 
| 144 | } | 148 | } | 
| 145 | bActive = false; | 149 | bActive = false; | 
| 146 | } | 150 | } | 
| 147 | 151 | ||
| 148 | size_t Bu::Socket::read( void *pBuf, size_t nBytes ) | 152 | size_t Bu::TcpSocket::read( void *pBuf, size_t nBytes ) | 
| 149 | { | 153 | { | 
| 150 | fd_set rfds; | 154 | fd_set rfds; | 
| 151 | FD_ZERO(&rfds); | 155 | FD_ZERO(&rfds); | 
| 152 | FD_SET(nSocket, &rfds); | 156 | FD_SET(nTcpSocket, &rfds); | 
| 153 | struct timeval tv = {0, 0}; | 157 | struct timeval tv = {0, 0}; | 
| 154 | if( bu_select( nSocket+1, &rfds, NULL, NULL, &tv ) < 0 ) | 158 | if( bu_select( nTcpSocket+1, &rfds, NULL, NULL, &tv ) < 0 ) | 
| 155 | { | 159 | { | 
| 156 | int iErr = errno; | 160 | int iErr = errno; | 
| 157 | close(); | 161 | close(); | 
| 158 | throw SocketException( SocketException::cRead, strerror(iErr) ); | 162 | throw TcpSocketException( TcpSocketException::cRead, strerror(iErr) ); | 
| 159 | } | 163 | } | 
| 160 | if( FD_ISSET( nSocket, &rfds ) || bBlocking ) | 164 | if( FD_ISSET( nTcpSocket, &rfds ) || bBlocking ) | 
| 161 | { | 165 | { | 
| 162 | int nRead = TEMP_FAILURE_RETRY( | 166 | int nRead = TEMP_FAILURE_RETRY( | 
| 163 | bu_recv( nSocket, (char *) pBuf, nBytes, 0 ) ); | 167 | bu_recv( nTcpSocket, (char *) pBuf, nBytes, 0 ) ); | 
| 164 | if( nRead == 0 ) | 168 | if( nRead == 0 ) | 
| 165 | { | 169 | { | 
| 166 | close(); | 170 | close(); | 
| 167 | throw SocketException( SocketException::cClosed, "Socket closed."); | 171 | throw TcpSocketException( TcpSocketException::cClosed, "TcpSocket closed."); | 
| 168 | } | 172 | } | 
| 169 | if( nRead < 0 ) | 173 | if( nRead < 0 ) | 
| 170 | { | 174 | { | 
| @@ -176,14 +180,14 @@ size_t Bu::Socket::read( void *pBuf, size_t nBytes ) | |||
| 176 | if( errno == ENETRESET || errno == ECONNRESET ) | 180 | if( errno == ENETRESET || errno == ECONNRESET ) | 
| 177 | { | 181 | { | 
| 178 | close(); | 182 | close(); | 
| 179 | throw SocketException( SocketException::cClosed, | 183 | throw TcpSocketException( TcpSocketException::cClosed, | 
| 180 | strerror(errno) ); | 184 | strerror(errno) ); | 
| 181 | } | 185 | } | 
| 182 | if( errno == EAGAIN ) | 186 | if( errno == EAGAIN ) | 
| 183 | return 0; | 187 | return 0; | 
| 184 | int iErr = errno; | 188 | int iErr = errno; | 
| 185 | close(); | 189 | close(); | 
| 186 | throw SocketException( SocketException::cRead, strerror(iErr) ); | 190 | throw TcpSocketException( TcpSocketException::cRead, strerror(iErr) ); | 
| 187 | #endif | 191 | #endif | 
| 188 | } | 192 | } | 
| 189 | return nRead; | 193 | return nRead; | 
| @@ -191,7 +195,7 @@ size_t Bu::Socket::read( void *pBuf, size_t nBytes ) | |||
| 191 | return 0; | 195 | return 0; | 
| 192 | } | 196 | } | 
| 193 | 197 | ||
| 194 | size_t Bu::Socket::read( void *pBuf, size_t nBytes, | 198 | size_t Bu::TcpSocket::read( void *pBuf, size_t nBytes, | 
| 195 | uint32_t nSec, uint32_t nUSec ) | 199 | uint32_t nSec, uint32_t nUSec ) | 
| 196 | { | 200 | { | 
| 197 | struct timeval tv; | 201 | struct timeval tv; | 
| @@ -199,7 +203,7 @@ size_t Bu::Socket::read( void *pBuf, size_t nBytes, | |||
| 199 | 203 | ||
| 200 | fd_set rfds; | 204 | fd_set rfds; | 
| 201 | FD_ZERO(&rfds); | 205 | FD_ZERO(&rfds); | 
| 202 | FD_SET(nSocket, &rfds); | 206 | FD_SET(nTcpSocket, &rfds); | 
| 203 | 207 | ||
| 204 | #ifdef WIN32 | 208 | #ifdef WIN32 | 
| 205 | DWORD dwStart = GetTickCount(); | 209 | DWORD dwStart = GetTickCount(); | 
| @@ -216,7 +220,7 @@ size_t Bu::Socket::read( void *pBuf, size_t nBytes, | |||
| 216 | { | 220 | { | 
| 217 | tv.tv_sec = nSec; | 221 | tv.tv_sec = nSec; | 
| 218 | tv.tv_usec = nUSec; | 222 | tv.tv_usec = nUSec; | 
| 219 | bu_select( nSocket+1, &rfds, NULL, NULL, &tv ); | 223 | bu_select( nTcpSocket+1, &rfds, NULL, NULL, &tv ); | 
| 220 | nRead += read( ((char *)pBuf)+nRead, nBytes-nRead ); | 224 | nRead += read( ((char *)pBuf)+nRead, nBytes-nRead ); | 
| 221 | if( nRead >= nBytes ) | 225 | if( nRead >= nBytes ) | 
| 222 | break; | 226 | break; | 
| @@ -235,13 +239,13 @@ size_t Bu::Socket::read( void *pBuf, size_t nBytes, | |||
| 235 | return nRead; | 239 | return nRead; | 
| 236 | } | 240 | } | 
| 237 | 241 | ||
| 238 | size_t Bu::Socket::write( const void *pBuf, size_t nBytes ) | 242 | size_t Bu::TcpSocket::write( const void *pBuf, size_t nBytes ) | 
| 239 | { | 243 | { | 
| 240 | //#ifdef WIN32 | 244 | //#ifdef WIN32 | 
| 241 | int nWrote = TEMP_FAILURE_RETRY( | 245 | int nWrote = TEMP_FAILURE_RETRY( | 
| 242 | bu_send( nSocket, (const char *) pBuf, nBytes, 0 ) ); | 246 | bu_send( nTcpSocket, (const char *) pBuf, nBytes, 0 ) ); | 
| 243 | //#else | 247 | //#else | 
| 244 | // int nWrote = TEMP_FAILURE_RETRY( ::write( nSocket, pBuf, nBytes ) ); | 248 | // int nWrote = TEMP_FAILURE_RETRY( ::write( nTcpSocket, pBuf, nBytes ) ); | 
| 245 | //#endif | 249 | //#endif | 
| 246 | if( nWrote < 0 ) | 250 | if( nWrote < 0 ) | 
| 247 | { | 251 | { | 
| @@ -252,19 +256,19 @@ size_t Bu::Socket::write( const void *pBuf, size_t nBytes ) | |||
| 252 | #else | 256 | #else | 
| 253 | if( errno == EAGAIN ) return 0; | 257 | if( errno == EAGAIN ) return 0; | 
| 254 | #endif | 258 | #endif | 
| 255 | throw SocketException( SocketException::cWrite, strerror(errno) ); | 259 | throw TcpSocketException( TcpSocketException::cWrite, strerror(errno) ); | 
| 256 | } | 260 | } | 
| 257 | return nWrote; | 261 | return nWrote; | 
| 258 | } | 262 | } | 
| 259 | 263 | ||
| 260 | size_t Bu::Socket::write( const void *pBuf, size_t nBytes, uint32_t nSec, uint32_t nUSec ) | 264 | size_t Bu::TcpSocket::write( const void *pBuf, size_t nBytes, uint32_t nSec, uint32_t nUSec ) | 
| 261 | { | 265 | { | 
| 262 | struct timeval tv; | 266 | struct timeval tv; | 
| 263 | size_t nWrote = 0; | 267 | size_t nWrote = 0; | 
| 264 | 268 | ||
| 265 | fd_set wfds; | 269 | fd_set wfds; | 
| 266 | FD_ZERO(&wfds); | 270 | FD_ZERO(&wfds); | 
| 267 | FD_SET(nSocket, &wfds); | 271 | FD_SET(nTcpSocket, &wfds); | 
| 268 | 272 | ||
| 269 | #ifdef WIN32 | 273 | #ifdef WIN32 | 
| 270 | DWORD dwStart = GetTickCount(); | 274 | DWORD dwStart = GetTickCount(); | 
| @@ -281,7 +285,7 @@ size_t Bu::Socket::write( const void *pBuf, size_t nBytes, uint32_t nSec, uint32 | |||
| 281 | { | 285 | { | 
| 282 | tv.tv_sec = nSec; | 286 | tv.tv_sec = nSec; | 
| 283 | tv.tv_usec = nUSec; | 287 | tv.tv_usec = nUSec; | 
| 284 | bu_select( nSocket+1, NULL, &wfds, NULL, &tv ); | 288 | bu_select( nTcpSocket+1, NULL, &wfds, NULL, &tv ); | 
| 285 | nWrote += write( ((char *)pBuf)+nWrote, nBytes-nWrote ); | 289 | nWrote += write( ((char *)pBuf)+nWrote, nBytes-nWrote ); | 
| 286 | if( nWrote >= nBytes ) | 290 | if( nWrote >= nBytes ) | 
| 287 | break; | 291 | break; | 
| @@ -300,101 +304,101 @@ size_t Bu::Socket::write( const void *pBuf, size_t nBytes, uint32_t nSec, uint32 | |||
| 300 | return nWrote; | 304 | return nWrote; | 
| 301 | } | 305 | } | 
| 302 | 306 | ||
| 303 | long Bu::Socket::tell() | 307 | long Bu::TcpSocket::tell() | 
| 304 | { | 308 | { | 
| 305 | throw UnsupportedException(); | 309 | throw UnsupportedException(); | 
| 306 | } | 310 | } | 
| 307 | 311 | ||
| 308 | void Bu::Socket::seek( long ) | 312 | void Bu::TcpSocket::seek( long ) | 
| 309 | { | 313 | { | 
| 310 | throw UnsupportedException(); | 314 | throw UnsupportedException(); | 
| 311 | } | 315 | } | 
| 312 | 316 | ||
| 313 | void Bu::Socket::setPos( long ) | 317 | void Bu::TcpSocket::setPos( long ) | 
| 314 | { | 318 | { | 
| 315 | throw UnsupportedException(); | 319 | throw UnsupportedException(); | 
| 316 | } | 320 | } | 
| 317 | 321 | ||
| 318 | void Bu::Socket::setPosEnd( long ) | 322 | void Bu::TcpSocket::setPosEnd( long ) | 
| 319 | { | 323 | { | 
| 320 | throw UnsupportedException(); | 324 | throw UnsupportedException(); | 
| 321 | } | 325 | } | 
| 322 | 326 | ||
| 323 | bool Bu::Socket::isEos() | 327 | bool Bu::TcpSocket::isEos() | 
| 324 | { | 328 | { | 
| 325 | return !bActive; | 329 | return !bActive; | 
| 326 | } | 330 | } | 
| 327 | 331 | ||
| 328 | bool Bu::Socket::canRead() | 332 | bool Bu::TcpSocket::canRead() | 
| 329 | { | 333 | { | 
| 330 | fd_set rfds; | 334 | fd_set rfds; | 
| 331 | FD_ZERO(&rfds); | 335 | FD_ZERO(&rfds); | 
| 332 | FD_SET(nSocket, &rfds); | 336 | FD_SET(nTcpSocket, &rfds); | 
| 333 | struct timeval tv = { 0, 0 }; | 337 | struct timeval tv = { 0, 0 }; | 
| 334 | int retval = bu_select( nSocket+1, &rfds, NULL, NULL, &tv ); | 338 | int retval = bu_select( nTcpSocket+1, &rfds, NULL, NULL, &tv ); | 
| 335 | if( retval == -1 ) | 339 | if( retval == -1 ) | 
| 336 | throw SocketException( | 340 | throw TcpSocketException( | 
| 337 | SocketException::cBadRead, | 341 | TcpSocketException::cBadRead, | 
| 338 | "Bad Read error" | 342 | "Bad Read error" | 
| 339 | ); | 343 | ); | 
| 340 | 344 | ||
| 341 | if( !FD_ISSET( nSocket, &rfds ) ) | 345 | if( !FD_ISSET( nTcpSocket, &rfds ) ) | 
| 342 | return false; | 346 | return false; | 
| 343 | return true; | 347 | return true; | 
| 344 | } | 348 | } | 
| 345 | 349 | ||
| 346 | bool Bu::Socket::canWrite() | 350 | bool Bu::TcpSocket::canWrite() | 
| 347 | { | 351 | { | 
| 348 | fd_set wfds; | 352 | fd_set wfds; | 
| 349 | FD_ZERO(&wfds); | 353 | FD_ZERO(&wfds); | 
| 350 | FD_SET(nSocket, &wfds); | 354 | FD_SET(nTcpSocket, &wfds); | 
| 351 | struct timeval tv = { 0, 0 }; | 355 | struct timeval tv = { 0, 0 }; | 
| 352 | int retval = bu_select( nSocket+1, NULL, &wfds, NULL, &tv ); | 356 | int retval = bu_select( nTcpSocket+1, NULL, &wfds, NULL, &tv ); | 
| 353 | if( retval == -1 ) | 357 | if( retval == -1 ) | 
| 354 | throw SocketException( | 358 | throw TcpSocketException( | 
| 355 | SocketException::cBadRead, | 359 | TcpSocketException::cBadRead, | 
| 356 | "Bad Read error" | 360 | "Bad Read error" | 
| 357 | ); | 361 | ); | 
| 358 | if( !FD_ISSET( nSocket, &wfds ) ) | 362 | if( !FD_ISSET( nTcpSocket, &wfds ) ) | 
| 359 | return false; | 363 | return false; | 
| 360 | return true; | 364 | return true; | 
| 361 | } | 365 | } | 
| 362 | 366 | ||
| 363 | bool Bu::Socket::isReadable() | 367 | bool Bu::TcpSocket::isReadable() | 
| 364 | { | 368 | { | 
| 365 | return true; | 369 | return true; | 
| 366 | } | 370 | } | 
| 367 | 371 | ||
| 368 | bool Bu::Socket::isWritable() | 372 | bool Bu::TcpSocket::isWritable() | 
| 369 | { | 373 | { | 
| 370 | return true; | 374 | return true; | 
| 371 | } | 375 | } | 
| 372 | 376 | ||
| 373 | bool Bu::Socket::isSeekable() | 377 | bool Bu::TcpSocket::isSeekable() | 
| 374 | { | 378 | { | 
| 375 | return false; | 379 | return false; | 
| 376 | } | 380 | } | 
| 377 | 381 | ||
| 378 | bool Bu::Socket::isBlocking() | 382 | bool Bu::TcpSocket::isBlocking() | 
| 379 | { | 383 | { | 
| 380 | #ifndef WIN32 | 384 | #ifndef WIN32 | 
| 381 | return ((fcntl( nSocket, F_GETFL, 0 ) & O_NONBLOCK) != O_NONBLOCK); | 385 | return ((fcntl( nTcpSocket, F_GETFL, 0 ) & O_NONBLOCK) != O_NONBLOCK); | 
| 382 | #else | 386 | #else | 
| 383 | return false; | 387 | return false; | 
| 384 | #endif | 388 | #endif | 
| 385 | } | 389 | } | 
| 386 | 390 | ||
| 387 | void Bu::Socket::setBlocking( bool bBlocking ) | 391 | void Bu::TcpSocket::setBlocking( bool bBlocking ) | 
| 388 | { | 392 | { | 
| 389 | this->bBlocking = bBlocking; | 393 | this->bBlocking = bBlocking; | 
| 390 | #ifndef WIN32 | 394 | #ifndef WIN32 | 
| 391 | if( bBlocking ) | 395 | if( bBlocking ) | 
| 392 | { | 396 | { | 
| 393 | fcntl( nSocket, F_SETFL, fcntl( nSocket, F_GETFL, 0 ) & (~O_NONBLOCK) ); | 397 | fcntl( nTcpSocket, F_SETFL, fcntl( nTcpSocket, F_GETFL, 0 ) & (~O_NONBLOCK) ); | 
| 394 | } | 398 | } | 
| 395 | else | 399 | else | 
| 396 | { | 400 | { | 
| 397 | fcntl( nSocket, F_SETFL, fcntl( nSocket, F_GETFL, 0 ) | O_NONBLOCK ); | 401 | fcntl( nTcpSocket, F_SETFL, fcntl( nTcpSocket, F_GETFL, 0 ) | O_NONBLOCK ); | 
| 398 | } | 402 | } | 
| 399 | #else | 403 | #else | 
| 400 | u_long iMode; | 404 | u_long iMode; | 
| @@ -408,39 +412,39 @@ void Bu::Socket::setBlocking( bool bBlocking ) | |||
| 408 | // socket based on the numerical value of iMode. | 412 | // socket based on the numerical value of iMode. | 
| 409 | // If iMode = 0, blocking is enabled; | 413 | // If iMode = 0, blocking is enabled; | 
| 410 | // If iMode != 0, non-blocking mode is enabled. | 414 | // If iMode != 0, non-blocking mode is enabled. | 
| 411 | bu_ioctlsocket(nSocket, FIONBIO, &iMode); | 415 | bu_ioctlsocket(nTcpSocket, FIONBIO, &iMode); | 
| 412 | #endif | 416 | #endif | 
| 413 | } | 417 | } | 
| 414 | 418 | ||
| 415 | void Bu::Socket::setSize( long ) | 419 | void Bu::TcpSocket::setSize( long ) | 
| 416 | { | 420 | { | 
| 417 | } | 421 | } | 
| 418 | 422 | ||
| 419 | void Bu::Socket::flush() | 423 | void Bu::TcpSocket::flush() | 
| 420 | { | 424 | { | 
| 421 | } | 425 | } | 
| 422 | 426 | ||
| 423 | bool Bu::Socket::isOpen() | 427 | bool Bu::TcpSocket::isOpen() | 
| 424 | { | 428 | { | 
| 425 | return bActive; | 429 | return bActive; | 
| 426 | } | 430 | } | 
| 427 | 431 | ||
| 428 | void Bu::Socket::setAddress() | 432 | void Bu::TcpSocket::setAddress() | 
| 429 | { | 433 | { | 
| 430 | struct sockaddr_in addr; | 434 | struct sockaddr_in addr; | 
| 431 | socklen_t len = sizeof(addr); | 435 | socklen_t len = sizeof(addr); | 
| 432 | addr.sin_family = AF_INET; | 436 | addr.sin_family = AF_INET; | 
| 433 | bu_getpeername( nSocket, (sockaddr *)(&addr), &len ); | 437 | bu_getpeername( nTcpSocket, (sockaddr *)(&addr), &len ); | 
| 434 | sAddress = bu_inet_ntoa( addr.sin_addr ); | 438 | sAddress = bu_inet_ntoa( addr.sin_addr ); | 
| 435 | } | 439 | } | 
| 436 | 440 | ||
| 437 | Bu::FString Bu::Socket::getAddress() const | 441 | Bu::FString Bu::TcpSocket::getAddress() const | 
| 438 | { | 442 | { | 
| 439 | return sAddress; | 443 | return sAddress; | 
| 440 | } | 444 | } | 
| 441 | 445 | ||
| 442 | Bu::Socket::operator int() const | 446 | Bu::TcpSocket::operator int() const | 
| 443 | { | 447 | { | 
| 444 | return nSocket; | 448 | return nTcpSocket; | 
| 445 | } | 449 | } | 
| 446 | 450 | ||
| diff --git a/src/socket.h b/src/tcpsocket.h index c8f78f0..3361e84 100644 --- a/src/socket.h +++ b/src/tcpsocket.h | |||
| @@ -5,8 +5,8 @@ | |||
| 5 | * terms of the license contained in the file LICENSE. | 5 | * terms of the license contained in the file LICENSE. | 
| 6 | */ | 6 | */ | 
| 7 | 7 | ||
| 8 | #ifndef BU_SOCKET_H | 8 | #ifndef BU_TCP_SOCKET_H | 
| 9 | #define BU_SOCKET_H | 9 | #define BU_TCP_SOCKET_H | 
| 10 | 10 | ||
| 11 | #include <stdint.h> | 11 | #include <stdint.h> | 
| 12 | 12 | ||
| @@ -16,7 +16,7 @@ | |||
| 16 | 16 | ||
| 17 | namespace Bu | 17 | namespace Bu | 
| 18 | { | 18 | { | 
| 19 | subExceptionDeclBegin( SocketException ); | 19 | subExceptionDeclBegin( TcpSocketException ); | 
| 20 | enum { | 20 | enum { | 
| 21 | cRead, | 21 | cRead, | 
| 22 | cWrite, | 22 | cWrite, | 
| @@ -40,28 +40,29 @@ namespace Bu | |||
| 40 | * Please note that there is a condition that will occur eventually (at | 40 | * Please note that there is a condition that will occur eventually (at | 
| 41 | * least on *nix systems) that will trigger a SIGPIPE condition. This | 41 | * least on *nix systems) that will trigger a SIGPIPE condition. This | 
| 42 | * will terminate your program immediately unless handled properly. Most | 42 | * will terminate your program immediately unless handled properly. Most | 
| 43 | * people doing any connections with Socket will want to put this in their | 43 | * people doing any connections with TcpSocket will want to put this in | 
| 44 | * program somewhere before they use it: | 44 | * their program somewhere before they use it: | 
| 45 | *@code | 45 | *@code | 
| 46 | #include <signal.h> | 46 | #include <signal.h> | 
| 47 | ... | 47 | ... | 
| 48 | ... | 48 | ... | 
| 49 | ... | 49 | ... | 
| 50 | sigset( SIGPIPE, SIG_IGN ); // do this before you use a Bu::Socket | 50 | sigset( SIGPIPE, SIG_IGN ); // do this before you use a Bu::TcpSocket | 
| 51 | @endcode | 51 | @endcode | 
| 52 | * When this is done, Bu::Socket will simply throw a broken pipe exception | 52 | * When this is done, Bu::TcpSocket will simply throw a broken pipe | 
| 53 | * just like every other error condition, allowing your program to handle | 53 | * exception just like every other error condition, allowing your program | 
| 54 | * it sanely. | 54 | * to handle it sanely. | 
| 55 | * | 55 | * | 
| 56 | *@ingroup Serving | 56 | *@ingroup Serving | 
| 57 | *@ingroup Streams | 57 | *@ingroup Streams | 
| 58 | */ | 58 | */ | 
| 59 | class Socket : public Stream | 59 | class TcpSocket : public Stream | 
| 60 | { | 60 | { | 
| 61 | public: | 61 | public: | 
| 62 | Socket( int nSocket ); | 62 | TcpSocket( int nTcpSocket ); | 
| 63 | Socket( const FString &sAddr, int nPort, int nTimeout=30 ); | 63 | TcpSocket( const FString &sAddr, int nPort, int nTimeout=30, | 
| 64 | virtual ~Socket(); | 64 | bool bBlocking=true ); | 
| 65 | virtual ~TcpSocket(); | ||
| 65 | 66 | ||
| 66 | virtual void close(); | 67 | virtual void close(); | 
| 67 | //virtual void read(); | 68 | //virtual void read(); | 
| @@ -101,9 +102,9 @@ namespace Bu | |||
| 101 | void setAddress(); | 102 | void setAddress(); | 
| 102 | 103 | ||
| 103 | #ifdef WIN32 | 104 | #ifdef WIN32 | 
| 104 | unsigned int nSocket; | 105 | unsigned int nTcpSocket; | 
| 105 | #else | 106 | #else | 
| 106 | int nSocket; | 107 | int nTcpSocket; | 
| 107 | #endif | 108 | #endif | 
| 108 | bool bActive; | 109 | bool bActive; | 
| 109 | bool bBlocking; | 110 | bool bBlocking; | 
| diff --git a/src/tests/sharedcore.cpp b/src/tests/sharedcore.cpp index 1d0c16e..9b0a0ec 100644 --- a/src/tests/sharedcore.cpp +++ b/src/tests/sharedcore.cpp | |||
| @@ -14,7 +14,7 @@ struct ShintCore | |||
| 14 | { | 14 | { | 
| 15 | int val; | 15 | int val; | 
| 16 | }; | 16 | }; | 
| 17 | class Shint : public Bu::SharedCore<struct ShintCore> | 17 | class Shint : public Bu::SharedCore<Shint, struct ShintCore> | 
| 18 | { | 18 | { | 
| 19 | public: | 19 | public: | 
| 20 | Shint() | 20 | Shint() | 
| diff --git a/src/tests/socketblock.cpp b/src/tests/socketblock.cpp index a1ea18d..793ef96 100644 --- a/src/tests/socketblock.cpp +++ b/src/tests/socketblock.cpp | |||
| @@ -6,8 +6,8 @@ | |||
| 6 | */ | 6 | */ | 
| 7 | 7 | ||
| 8 | #include "bu/ito.h" | 8 | #include "bu/ito.h" | 
| 9 | #include "bu/socket.h" | 9 | #include "bu/tcpsocket.h" | 
| 10 | #include "bu/serversocket.h" | 10 | #include "bu/tcpserversocket.h" | 
| 11 | #include <stdio.h> | 11 | #include <stdio.h> | 
| 12 | #include <unistd.h> | 12 | #include <unistd.h> | 
| 13 | 13 | ||
| @@ -21,7 +21,7 @@ public: | |||
| 21 | 21 | ||
| 22 | virtual void run() | 22 | virtual void run() | 
| 23 | { | 23 | { | 
| 24 | Bu::Socket c = s.accept( 45, 0 ); | 24 | Bu::TcpSocket c = s.accept( 45, 0 ); | 
| 25 | printf("TstServer: Accetped connection.\n"); fflush( stdout ); | 25 | printf("TstServer: Accetped connection.\n"); fflush( stdout ); | 
| 26 | 26 | ||
| 27 | sleep( 1 ); | 27 | sleep( 1 ); | 
| @@ -35,7 +35,7 @@ public: | |||
| 35 | c.close(); | 35 | c.close(); | 
| 36 | } | 36 | } | 
| 37 | 37 | ||
| 38 | Bu::ServerSocket s; | 38 | Bu::TcpServerSocket s; | 
| 39 | }; | 39 | }; | 
| 40 | 40 | ||
| 41 | int main() | 41 | int main() | 
| @@ -45,7 +45,7 @@ int main() | |||
| 45 | ts.start(); | 45 | ts.start(); | 
| 46 | 46 | ||
| 47 | printf("main: Connecting to server.\n"); fflush( stdout ); | 47 | printf("main: Connecting to server.\n"); fflush( stdout ); | 
| 48 | Bu::Socket s( "localhost", 55678 ); | 48 | Bu::TcpSocket s( "localhost", 55678 ); | 
| 49 | 49 | ||
| 50 | printf("main: Sending 4 bytes.\n"); fflush( stdout ); | 50 | printf("main: Sending 4 bytes.\n"); fflush( stdout ); | 
| 51 | s.write( "aoeu", 4 ); | 51 | s.write( "aoeu", 4 ); | 
| diff --git a/src/tests/socketbreak.cpp b/src/tests/socketbreak.cpp index 8339630..7d3c71a 100644 --- a/src/tests/socketbreak.cpp +++ b/src/tests/socketbreak.cpp | |||
| @@ -5,17 +5,17 @@ | |||
| 5 | * terms of the license contained in the file LICENSE. | 5 | * terms of the license contained in the file LICENSE. | 
| 6 | */ | 6 | */ | 
| 7 | 7 | ||
| 8 | #include "bu/serversocket.h" | 8 | #include "bu/tcpserversocket.h" | 
| 9 | #include "bu/socket.h" | 9 | #include "bu/tcpsocket.h" | 
| 10 | #include <unistd.h> | 10 | #include <unistd.h> | 
| 11 | 11 | ||
| 12 | int main() | 12 | int main() | 
| 13 | { | 13 | { | 
| 14 | Bu::ServerSocket sSrv( 9987 ); | 14 | Bu::TcpServerSocket sSrv( 9987 ); | 
| 15 | 15 | ||
| 16 | Bu::Socket sSend("localhost", 9987 ); | 16 | Bu::TcpSocket sSend("localhost", 9987 ); | 
| 17 | 17 | ||
| 18 | Bu::Socket sRecv( sSrv.accept() ); | 18 | Bu::TcpSocket sRecv( sSrv.accept() ); | 
| 19 | 19 | ||
| 20 | printf("Connected sockets.\n"); | 20 | printf("Connected sockets.\n"); | 
| 21 | 21 | ||
| diff --git a/src/tests/socket.cpp b/src/tests/tcpsocket.cpp index ba4e9b9..30dd22f 100644 --- a/src/tests/socket.cpp +++ b/src/tests/tcpsocket.cpp | |||
| @@ -5,7 +5,7 @@ | |||
| 5 | * terms of the license contained in the file LICENSE. | 5 | * terms of the license contained in the file LICENSE. | 
| 6 | */ | 6 | */ | 
| 7 | 7 | ||
| 8 | #include <bu/socket.h> | 8 | #include <bu/tcpsocket.h> | 
| 9 | #include <bu/sio.h> | 9 | #include <bu/sio.h> | 
| 10 | 10 | ||
| 11 | #include <sys/time.h> | 11 | #include <sys/time.h> | 
| @@ -17,7 +17,7 @@ bool isUp() | |||
| 17 | { | 17 | { | 
| 18 | try | 18 | try | 
| 19 | { | 19 | { | 
| 20 | Socket s("xagasoft.com", 9898, 1 ); | 20 | TcpSocket s("xagasoft.com", 9898, 1 ); | 
| 21 | 21 | ||
| 22 | char buf[5]; | 22 | char buf[5]; | 
| 23 | buf[s.read(buf, 2, 1, 0)] = '\0'; | 23 | buf[s.read(buf, 2, 1, 0)] = '\0'; | 
| diff --git a/src/tools/bnfcompile.cpp b/src/tools/bnfcompile.cpp new file mode 100644 index 0000000..16e75a5 --- /dev/null +++ b/src/tools/bnfcompile.cpp | |||
| @@ -0,0 +1,410 @@ | |||
| 1 | #include <bu/sio.h> | ||
| 2 | #include <bu/lexer.h> | ||
| 3 | #include <bu/parser.h> | ||
| 4 | #include <bu/file.h> | ||
| 5 | #include <bu/queuebuf.h> | ||
| 6 | |||
| 7 | using namespace Bu; | ||
| 8 | |||
| 9 | enum TokenType | ||
| 10 | { | ||
| 11 | tokIdentifier, | ||
| 12 | tokColon, | ||
| 13 | tokOr, | ||
| 14 | tokSemiColon, | ||
| 15 | tokTokens, | ||
| 16 | tokEquals, | ||
| 17 | tokOpenCurly, | ||
| 18 | tokCloseCurly, | ||
| 19 | tokOpenSquare, | ||
| 20 | tokCloseSquare, | ||
| 21 | |||
| 22 | tokEos=-1 | ||
| 23 | }; | ||
| 24 | |||
| 25 | class BnfLexer : public Lexer | ||
| 26 | { | ||
| 27 | public: | ||
| 28 | BnfLexer( Stream &rSrc ) : | ||
| 29 | rSrc( rSrc ) | ||
| 30 | { | ||
| 31 | } | ||
| 32 | |||
| 33 | virtual ~BnfLexer() | ||
| 34 | { | ||
| 35 | } | ||
| 36 | |||
| 37 | virtual Token *nextToken() | ||
| 38 | { | ||
| 39 | char cBuf; | ||
| 40 | |||
| 41 | for(;;) | ||
| 42 | { | ||
| 43 | if( qbIn.getSize() == 0 ) | ||
| 44 | { | ||
| 45 | char buf[4096]; | ||
| 46 | qbIn.write( buf, rSrc.read( buf, 4096 ) ); | ||
| 47 | |||
| 48 | if( rSrc.isEos() && qbIn.getSize() == 0 ) | ||
| 49 | return new Token( tokEos ); | ||
| 50 | } | ||
| 51 | qbIn.peek( &cBuf, 1 ); | ||
| 52 | if( (cBuf >= 'a' && cBuf <= 'z') || | ||
| 53 | (cBuf >= 'A' && cBuf <= 'Z') || | ||
| 54 | (cBuf >= '0' && cBuf <= '9') || | ||
| 55 | cBuf == '_' ) | ||
| 56 | { | ||
| 57 | sBuf.append( cBuf ); | ||
| 58 | qbIn.seek( 1 ); | ||
| 59 | } | ||
| 60 | else if( sBuf.isSet() ) | ||
| 61 | { | ||
| 62 | if( sBuf == "tokens" ) | ||
| 63 | { | ||
| 64 | sBuf.clear(); | ||
| 65 | return new Token( tokTokens ); | ||
| 66 | } | ||
| 67 | else | ||
| 68 | { | ||
| 69 | Token *pRet = new Token( tokIdentifier, sBuf ); | ||
| 70 | sBuf.clear(); | ||
| 71 | return pRet; | ||
| 72 | } | ||
| 73 | } | ||
| 74 | else | ||
| 75 | { | ||
| 76 | switch( cBuf ) | ||
| 77 | { | ||
| 78 | case ' ': | ||
| 79 | case '\t': | ||
| 80 | case '\n': | ||
| 81 | case '\r': | ||
| 82 | qbIn.seek( 1 ); | ||
| 83 | continue; | ||
| 84 | |||
| 85 | case ':': | ||
| 86 | qbIn.seek( 1 ); | ||
| 87 | return new Token( tokColon ); | ||
| 88 | |||
| 89 | case ';': | ||
| 90 | qbIn.seek( 1 ); | ||
| 91 | return new Token( tokSemiColon ); | ||
| 92 | |||
| 93 | case '|': | ||
| 94 | qbIn.seek( 1 ); | ||
| 95 | return new Token( tokOr ); | ||
| 96 | |||
| 97 | case '=': | ||
| 98 | qbIn.seek( 1 ); | ||
| 99 | return new Token( tokEquals ); | ||
| 100 | |||
| 101 | case '[': | ||
| 102 | qbIn.seek( 1 ); | ||
| 103 | return new Token( tokOpenSquare ); | ||
| 104 | |||
| 105 | case ']': | ||
| 106 | qbIn.seek( 1 ); | ||
| 107 | return new Token( tokCloseSquare ); | ||
| 108 | |||
| 109 | case '{': | ||
| 110 | qbIn.seek( 1 ); | ||
| 111 | return new Token( tokOpenCurly ); | ||
| 112 | |||
| 113 | case '}': | ||
| 114 | qbIn.seek( 1 ); | ||
| 115 | return new Token( tokCloseCurly ); | ||
| 116 | |||
| 117 | default: | ||
| 118 | throw ExceptionBase("Unexpected character '%c'.", | ||
| 119 | cBuf ); | ||
| 120 | break; | ||
| 121 | } | ||
| 122 | } | ||
| 123 | } | ||
| 124 | } | ||
| 125 | |||
| 126 | virtual FString tokenToString( const Token &t ) | ||
| 127 | { | ||
| 128 | switch( (TokenType)t.iToken ) | ||
| 129 | { | ||
| 130 | case tokIdentifier: return "tokIdentifier"; | ||
| 131 | case tokColon: return "tokColon"; | ||
| 132 | case tokOr: return "tokOr"; | ||
| 133 | case tokSemiColon: return "tokSemiColon"; | ||
| 134 | case tokTokens: return "tokTokens"; | ||
| 135 | case tokEquals: return "tokEquals"; | ||
| 136 | case tokOpenCurly: return "tokOpenCurly"; | ||
| 137 | case tokCloseCurly: return "tokCloseCurly"; | ||
| 138 | case tokOpenSquare: return "tokOpenSquare"; | ||
| 139 | case tokCloseSquare: return "tokCloseSquare"; | ||
| 140 | case tokEos: return "tokEos"; | ||
| 141 | } | ||
| 142 | |||
| 143 | return "???"; | ||
| 144 | } | ||
| 145 | |||
| 146 | private: | ||
| 147 | Stream &rSrc; | ||
| 148 | QueueBuf qbIn; | ||
| 149 | FString sBuf; | ||
| 150 | }; | ||
| 151 | |||
| 152 | class BnfParser | ||
| 153 | { | ||
| 154 | public: | ||
| 155 | BnfParser( BnfLexer &l ) : | ||
| 156 | l( l ), | ||
| 157 | pCur( NULL ), | ||
| 158 | iLastToken( 0 ) | ||
| 159 | { | ||
| 160 | } | ||
| 161 | |||
| 162 | virtual ~BnfParser() | ||
| 163 | { | ||
| 164 | delete pCur; | ||
| 165 | pCur = NULL; | ||
| 166 | } | ||
| 167 | |||
| 168 | void parse() | ||
| 169 | { | ||
| 170 | for(;;) | ||
| 171 | { | ||
| 172 | next(); | ||
| 173 | switch( pCur->iToken ) | ||
| 174 | { | ||
| 175 | case tokTokens: | ||
| 176 | tokens(); | ||
| 177 | break; | ||
| 178 | |||
| 179 | case tokIdentifier: | ||
| 180 | nonTerminal(); | ||
| 181 | break; | ||
| 182 | |||
| 183 | case tokEos: | ||
| 184 | return; | ||
| 185 | break; | ||
| 186 | |||
| 187 | default: | ||
| 188 | tokenError("tokTokens, tokIdentifier, or tokEos"); | ||
| 189 | } | ||
| 190 | } | ||
| 191 | } | ||
| 192 | |||
| 193 | private: | ||
| 194 | void tokens() | ||
| 195 | { | ||
| 196 | next(); | ||
| 197 | if( pCur->iToken != tokEquals ) | ||
| 198 | tokenError("tokEquals"); | ||
| 199 | for(;;) | ||
| 200 | { | ||
| 201 | next(); | ||
| 202 | if( pCur->iToken == tokIdentifier ) | ||
| 203 | { | ||
| 204 | hTokens.insert( pCur->vExtra.get<Bu::FString>(), ++iLastToken ); | ||
| 205 | sio << "Added token[" << iLastToken << "]: " | ||
| 206 | << pCur->vExtra.get<Bu::FString>() << sio.nl; | ||
| 207 | } | ||
| 208 | else if( pCur->iToken == tokSemiColon ) | ||
| 209 | break; | ||
| 210 | else | ||
| 211 | tokenError("tokIdentifier or tokSemiColon"); | ||
| 212 | } | ||
| 213 | } | ||
| 214 | |||
| 215 | void nonTerminal() | ||
| 216 | { | ||
| 217 | Bu::FString sNtName = pCur->vExtra.get<Bu::FString>(); | ||
| 218 | Parser::NonTerminal nt; | ||
| 219 | p.addNonTerminal( sNtName ); | ||
| 220 | sio.incIndent(); | ||
| 221 | sio << "Created non-terminal: " << sNtName << sio.nl; | ||
| 222 | |||
| 223 | next(); | ||
| 224 | if( pCur->iToken != tokColon ) | ||
| 225 | tokenError("tokColon"); | ||
| 226 | production( nt ); | ||
| 227 | for(;;) | ||
| 228 | { | ||
| 229 | switch( pCur->iToken ) | ||
| 230 | { | ||
| 231 | case tokOr: | ||
| 232 | production( nt ); | ||
| 233 | break; | ||
| 234 | |||
| 235 | case tokSemiColon: | ||
| 236 | p.setNonTerminal( sNtName, nt ); | ||
| 237 | sio.decIndent(); | ||
| 238 | sio << "Closing non-terminal." << sio.nl; | ||
| 239 | return; | ||
| 240 | |||
| 241 | default: | ||
| 242 | tokenError("tkOr or tokSemiColon"); | ||
| 243 | break; | ||
| 244 | } | ||
| 245 | } | ||
| 246 | } | ||
| 247 | |||
| 248 | void production( Parser::NonTerminal &nt ) | ||
| 249 | { | ||
| 250 | sio.incIndent(); | ||
| 251 | sio << "Adding new production:" << sio.nl; | ||
| 252 | Parser::Production pr; | ||
| 253 | bool bAnything = false; | ||
| 254 | for(;;) | ||
| 255 | { | ||
| 256 | next(); | ||
| 257 | switch( pCur->iToken ) | ||
| 258 | { | ||
| 259 | case tokIdentifier: | ||
| 260 | { | ||
| 261 | const Bu::FString &sName = | ||
| 262 | pCur->vExtra.get<Bu::FString>(); | ||
| 263 | if( hTokens.has( sName ) ) | ||
| 264 | { | ||
| 265 | pr.append( | ||
| 266 | Parser::State( | ||
| 267 | Parser::State::typeTerminal, | ||
| 268 | hTokens.get( sName ) | ||
| 269 | ) | ||
| 270 | ); | ||
| 271 | sio << "Added terminal " << sName << sio.nl; | ||
| 272 | } | ||
| 273 | else | ||
| 274 | { | ||
| 275 | if( !p.hasNonTerminal( sName ) ) | ||
| 276 | { | ||
| 277 | p.addNonTerminal( sName ); | ||
| 278 | } | ||
| 279 | pr.append( | ||
| 280 | Parser::State( | ||
| 281 | Parser::State::typeNonTerminal, | ||
| 282 | p.getNonTerminalId( sName ) | ||
| 283 | ) | ||
| 284 | ); | ||
| 285 | sio << "Added non-terminal " << sName << sio.nl; | ||
| 286 | } | ||
| 287 | } | ||
| 288 | break; | ||
| 289 | |||
| 290 | case tokOpenSquare: | ||
| 291 | { | ||
| 292 | next(); | ||
| 293 | if( pCur->iToken != tokIdentifier ) | ||
| 294 | tokenError("tokIdentifier"); | ||
| 295 | Bu::FString sName = | ||
| 296 | pCur->vExtra.get<Bu::FString>(); | ||
| 297 | next(); | ||
| 298 | if( pCur->iToken != tokCloseSquare ) | ||
| 299 | tokenError("tokCloseSquare"); | ||
| 300 | |||
| 301 | if( !hTokens.has( sName ) ) | ||
| 302 | throw ExceptionBase("Only token names may be " | ||
| 303 | "enclosed in square brackets."); | ||
| 304 | |||
| 305 | pr.append( | ||
| 306 | Parser::State( | ||
| 307 | Parser::State::typeTerminalPush, | ||
| 308 | hTokens.get( sName ) | ||
| 309 | ) | ||
| 310 | ); | ||
| 311 | sio << "Added terminal-push " << sName << sio.nl; | ||
| 312 | } | ||
| 313 | break; | ||
| 314 | |||
| 315 | case tokOpenCurly: | ||
| 316 | { | ||
| 317 | next(); | ||
| 318 | if( pCur->iToken != tokIdentifier ) | ||
| 319 | tokenError("tokIdentifier"); | ||
| 320 | Bu::FString sName = | ||
| 321 | pCur->vExtra.get<Bu::FString>(); | ||
| 322 | next(); | ||
| 323 | if( pCur->iToken != tokCloseCurly ) | ||
| 324 | tokenError("tokCloseCurly"); | ||
| 325 | |||
| 326 | if( !p.hasReduction( sName ) ) | ||
| 327 | p.addReduction( sName ); | ||
| 328 | |||
| 329 | pr.append( | ||
| 330 | Parser::State( | ||
| 331 | Parser::State::typeReduction, | ||
| 332 | p.getReductionId( sName ) | ||
| 333 | ) | ||
| 334 | ); | ||
| 335 | sio << "Added reduction " << sName << sio.nl; | ||
| 336 | } | ||
| 337 | break; | ||
| 338 | |||
| 339 | case tokOr: | ||
| 340 | case tokSemiColon: | ||
| 341 | if( bAnything ) | ||
| 342 | { | ||
| 343 | nt.addProduction( pr ); | ||
| 344 | sio.decIndent(); | ||
| 345 | sio << "Closing production." << sio.nl; | ||
| 346 | } | ||
| 347 | else | ||
| 348 | { | ||
| 349 | nt.setCanSkip(); | ||
| 350 | sio.decIndent(); | ||
| 351 | sio << "Closing empty production." << sio.nl; | ||
| 352 | } | ||
| 353 | return; | ||
| 354 | |||
| 355 | default: | ||
| 356 | tokenError("tokIdentifier, tokOpenSquare, tokOr, " | ||
| 357 | "tokOpenCurly, or tokSemiColon"); | ||
| 358 | } | ||
| 359 | } | ||
| 360 | } | ||
| 361 | |||
| 362 | private: | ||
| 363 | void next() | ||
| 364 | { | ||
| 365 | delete pCur; | ||
| 366 | pCur = l.nextToken(); | ||
| 367 | } | ||
| 368 | |||
| 369 | void tokenError( const FString &s ) | ||
| 370 | { | ||
| 371 | throw ExceptionBase( ("Expected " + s + " but found " | ||
| 372 | + l.tokenToString( *pCur ) + ".").getStr() ); | ||
| 373 | } | ||
| 374 | |||
| 375 | private: | ||
| 376 | typedef Bu::Hash<Bu::FString, int> TokenHash; | ||
| 377 | TokenHash hTokens; | ||
| 378 | BnfLexer &l; | ||
| 379 | BnfLexer::Token *pCur; | ||
| 380 | int iLastToken; | ||
| 381 | Parser p; | ||
| 382 | }; | ||
| 383 | |||
| 384 | int main( int argc, char *argv[] ) | ||
| 385 | { | ||
| 386 | File fIn( argv[1], File::Read ); | ||
| 387 | |||
| 388 | BnfLexer bl( fIn ); | ||
| 389 | BnfParser parser( bl ); | ||
| 390 | |||
| 391 | parser.parse(); | ||
| 392 | |||
| 393 | /* | ||
| 394 | for(;;) | ||
| 395 | { | ||
| 396 | Lexer::Token *pTok = bl.nextToken(); | ||
| 397 | sio << bl.tokenToString(*pTok); | ||
| 398 | if( pTok->vExtra.isSet() ) | ||
| 399 | { | ||
| 400 | sio << " - " << pTok->vExtra; | ||
| 401 | } | ||
| 402 | sio << sio.nl; | ||
| 403 | if( pTok->iToken == tokEos ) | ||
| 404 | break; | ||
| 405 | } | ||
| 406 | */ | ||
| 407 | |||
| 408 | return 0; | ||
| 409 | } | ||
| 410 | |||
| diff --git a/src/tools/myriadfs.cpp b/src/tools/myriadfs.cpp new file mode 100644 index 0000000..66955a5 --- /dev/null +++ b/src/tools/myriadfs.cpp | |||
| @@ -0,0 +1,38 @@ | |||
| 1 | #define FUSE_USE_VERSION 26 | ||
| 2 | |||
| 3 | #include <fuse.h> | ||
| 4 | |||
| 5 | #include <string.h> | ||
| 6 | |||
| 7 | extern "C" { | ||
| 8 | static int myriadfs_getattr( const char *sPath, struct stat *stbuf ) | ||
| 9 | { | ||
| 10 | |||
| 11 | } | ||
| 12 | |||
| 13 | static int myriadfs_readdir( const char *sPath, void *buf, | ||
| 14 | fuse_fill_dir_t filler, off_t offset, struct fuse_file_info *fi ) | ||
| 15 | { | ||
| 16 | } | ||
| 17 | |||
| 18 | static int myriadfs_open( const char *sPath, struct fuse_file_info *fi ) | ||
| 19 | { | ||
| 20 | } | ||
| 21 | |||
| 22 | static int myriadfs_read( const char *sPath, char *buf, size_t iSize, | ||
| 23 | off_t iOffset, struct fuse_file_info *fi ) | ||
| 24 | { | ||
| 25 | } | ||
| 26 | |||
| 27 | static struct fuse_operations myriadfs_oper; | ||
| 28 | |||
| 29 | int main( int argc, char *argv[] ) | ||
| 30 | { | ||
| 31 | memset( &myriadfs_oper, sizeof(myriadfs_oper), 0 ); | ||
| 32 | myriadfs_oper.getattr = myriadfs_getattr; | ||
| 33 | myriadfs_oper.readdir = myriadfs_readdir; | ||
| 34 | myriadfs_oper.open = myriadfs_open; | ||
| 35 | myriadfs_oper.read = myriadfs_read; | ||
| 36 | return fuse_main( argc, argv, &myriadfs_oper, NULL ); | ||
| 37 | } | ||
| 38 | } | ||
| diff --git a/src/tools/parser.cpp b/src/tools/parser.cpp index 76d4a72..7933f31 100644 --- a/src/tools/parser.cpp +++ b/src/tools/parser.cpp | |||
| @@ -151,10 +151,43 @@ private: | |||
| 151 | 151 | ||
| 152 | void redAdd( Bu::Parser &p ) | 152 | void redAdd( Bu::Parser &p ) | 
| 153 | { | 153 | { | 
| 154 | Lexer::Token *a = p.popToken(); | ||
| 155 | Lexer::Token *b = p.popToken(); | ||
| 156 | |||
| 157 | sio << "Add! " << b->vExtra.get<double>() << " + " | ||
| 158 | << a->vExtra.get<double>() << sio.nl; | ||
| 159 | |||
| 160 | Lexer::Token *c = new Lexer::Token( tokNumber, | ||
| 161 | b->vExtra.get<double>() + a->vExtra.get<double>() | ||
| 162 | ); | ||
| 163 | p.pushToken( c ); | ||
| 164 | |||
| 165 | delete a; | ||
| 166 | delete b; | ||
| 167 | } | ||
| 168 | |||
| 169 | void redSubtract( Bu::Parser &p ) | ||
| 170 | { | ||
| 171 | Lexer::Token *a = p.popToken(); | ||
| 172 | Lexer::Token *b = p.popToken(); | ||
| 173 | |||
| 174 | sio << "Subtract! " << b->vExtra.get<double>() << " - " | ||
| 175 | << a->vExtra.get<double>() << sio.nl; | ||
| 176 | |||
| 177 | Lexer::Token *c = new Lexer::Token( tokNumber, | ||
| 178 | b->vExtra.get<double>() - a->vExtra.get<double>() | ||
| 179 | ); | ||
| 180 | p.pushToken( c ); | ||
| 181 | |||
| 182 | delete a; | ||
| 183 | delete b; | ||
| 154 | } | 184 | } | 
| 155 | 185 | ||
| 156 | void redPrint( Bu::Parser &p ) | 186 | void redPrint( Bu::Parser &p ) | 
| 157 | { | 187 | { | 
| 188 | Lexer::Token *a = p.popToken(); | ||
| 189 | sio << "Print! = " << a->vExtra.get<double>() << sio.nl; | ||
| 190 | delete a; | ||
| 158 | } | 191 | } | 
| 159 | 192 | ||
| 160 | /* Basic grammer example: | 193 | /* Basic grammer example: | 
| @@ -170,14 +203,15 @@ void redPrint( Bu::Parser &p ) | |||
| 170 | * The problem is, that we can't actually make something left hand recursive, | 203 | * The problem is, that we can't actually make something left hand recursive, | 
| 171 | * so we break it into two exprs: | 204 | * so we break it into two exprs: | 
| 172 | * | 205 | * | 
| 173 | * expr': '(' expr ')' | 206 | * expr-sub1: '(' expr ')' | 
| 174 | * | NUMBER | 207 | * | NUMBER | 
| 175 | * ; | 208 | * ; | 
| 176 | * | 209 | * | 
| 177 | * expr: expr' expr'' | 210 | * expr: expr-sub1 expr-sub2 | 
| 178 | * ; | 211 | * ; | 
| 179 | * | 212 | * | 
| 180 | * expr'': '+' expr | 213 | * expr-sub2: '+' expr | 
| 214 | * | '-' expr | ||
| 181 | * | | 215 | * | | 
| 182 | * ; | 216 | * ; | 
| 183 | * | 217 | * | 
| @@ -191,8 +225,8 @@ int main( int argc, char *argv[] ) | |||
| 191 | Parser p; | 225 | Parser p; | 
| 192 | 226 | ||
| 193 | p.addNonTerminal("expr"); | 227 | p.addNonTerminal("expr"); | 
| 194 | p.addNonTerminal("expr'"); | 228 | p.addNonTerminal("expr-sub1"); | 
| 195 | p.addNonTerminal("expr''"); | 229 | p.addNonTerminal("expr-sub2"); | 
| 196 | { | 230 | { | 
| 197 | Parser::NonTerminal nt; | 231 | Parser::NonTerminal nt; | 
| 198 | nt.addProduction( | 232 | nt.addProduction( | 
| @@ -215,10 +249,28 @@ int main( int argc, char *argv[] ) | |||
| 215 | ); | 249 | ); | 
| 216 | nt.addProduction( | 250 | nt.addProduction( | 
| 217 | Parser::Production( | 251 | Parser::Production( | 
| 252 | Parser::State( | ||
| 253 | Parser::State::typeTerminal, | ||
| 254 | tokMinus | ||
| 255 | ) | ||
| 256 | ).append( | ||
| 257 | Parser::State( | ||
| 258 | Parser::State::typeNonTerminal, | ||
| 259 | p.getNonTerminalId("expr") | ||
| 260 | ) | ||
| 261 | ).append( | ||
| 262 | Parser::State( | ||
| 263 | Parser::State::typeReduction, | ||
| 264 | p.addReduction("subtract") | ||
| 265 | ) | ||
| 266 | ) | ||
| 267 | ); | ||
| 268 | nt.addProduction( | ||
| 269 | Parser::Production( | ||
| 218 | ) | 270 | ) | 
| 219 | ); | 271 | ); | 
| 220 | nt.setCanSkip(); | 272 | nt.setCanSkip(); | 
| 221 | p.setNonTerminal("expr''", nt ); | 273 | p.setNonTerminal("expr-sub2", nt ); | 
| 222 | } | 274 | } | 
| 223 | { | 275 | { | 
| 224 | Parser::NonTerminal nt; | 276 | Parser::NonTerminal nt; | 
| @@ -230,7 +282,25 @@ int main( int argc, char *argv[] ) | |||
| 230 | ) | 282 | ) | 
| 231 | ) | 283 | ) | 
| 232 | ); | 284 | ); | 
| 233 | p.setNonTerminal("expr'", nt ); | 285 | nt.addProduction( | 
| 286 | Parser::Production( | ||
| 287 | Parser::State( | ||
| 288 | Parser::State::typeTerminal, | ||
| 289 | tokOpenParen | ||
| 290 | ) | ||
| 291 | ).append( | ||
| 292 | Parser::State( | ||
| 293 | Parser::State::typeNonTerminal, | ||
| 294 | p.getNonTerminalId("expr") | ||
| 295 | ) | ||
| 296 | ).append( | ||
| 297 | Parser::State( | ||
| 298 | Parser::State::typeTerminal, | ||
| 299 | tokCloseParen | ||
| 300 | ) | ||
| 301 | ) | ||
| 302 | ); | ||
| 303 | p.setNonTerminal("expr-sub1", nt ); | ||
| 234 | } | 304 | } | 
| 235 | { | 305 | { | 
| 236 | Parser::NonTerminal nt; | 306 | Parser::NonTerminal nt; | 
| @@ -238,12 +308,12 @@ int main( int argc, char *argv[] ) | |||
| 238 | Parser::Production( | 308 | Parser::Production( | 
| 239 | Parser::State( | 309 | Parser::State( | 
| 240 | Parser::State::typeNonTerminal, | 310 | Parser::State::typeNonTerminal, | 
| 241 | p.getNonTerminalId("expr'") | 311 | p.getNonTerminalId("expr-sub1") | 
| 242 | ) | 312 | ) | 
| 243 | ).append( | 313 | ).append( | 
| 244 | Parser::State( | 314 | Parser::State( | 
| 245 | Parser::State::typeNonTerminal, | 315 | Parser::State::typeNonTerminal, | 
| 246 | p.getNonTerminalId("expr''") | 316 | p.getNonTerminalId("expr-sub2") | 
| 247 | ) | 317 | ) | 
| 248 | ) | 318 | ) | 
| 249 | ); | 319 | ); | 
| @@ -275,6 +345,7 @@ int main( int argc, char *argv[] ) | |||
| 275 | p.setRootNonTerminal("input"); | 345 | p.setRootNonTerminal("input"); | 
| 276 | 346 | ||
| 277 | p.setReduction("add", Bu::slot( &redAdd ) ); | 347 | p.setReduction("add", Bu::slot( &redAdd ) ); | 
| 348 | p.setReduction("subtract", Bu::slot( &redSubtract ) ); | ||
| 278 | p.setReduction("print", Bu::slot( &redPrint ) ); | 349 | p.setReduction("print", Bu::slot( &redPrint ) ); | 
| 279 | 350 | ||
| 280 | p.pushLexer( new MathLexer( fIn ) ); | 351 | p.pushLexer( new MathLexer( fIn ) ); | 
| diff --git a/src/unit/fstring.unit b/src/unit/fstring.unit index 00b6eed..c6d7414 100644 --- a/src/unit/fstring.unit +++ b/src/unit/fstring.unit | |||
| @@ -339,4 +339,13 @@ suite FString | |||
| 339 | a = "This is a test."; | 339 | a = "This is a test."; | 
| 340 | unitTest( a.replace("i", "ooo") == "Thooos ooos a test." ); | 340 | unitTest( a.replace("i", "ooo") == "Thooos ooos a test." ); | 
| 341 | } | 341 | } | 
| 342 | |||
| 343 | test coreDerefBug1 | ||
| 344 | { | ||
| 345 | Bu::FString a, b; | ||
| 346 | a = "bob"; | ||
| 347 | a.setSize( 0 ); | ||
| 348 | b = a; | ||
| 349 | b.getStr(); | ||
| 350 | } | ||
| 342 | } | 351 | } | 
| diff --git a/src/unit/hash.unit b/src/unit/hash.unit index e3d7e42..124b074 100644 --- a/src/unit/hash.unit +++ b/src/unit/hash.unit | |||
| @@ -84,4 +84,17 @@ suite Hash | |||
| 84 | printf(" - \"%s\" = %d\n", i.getKey().getStr(), i.getValue() ); | 84 | printf(" - \"%s\" = %d\n", i.getKey().getStr(), i.getValue() ); | 
| 85 | } */ | 85 | } */ | 
| 86 | } | 86 | } | 
| 87 | |||
| 88 | test copy | ||
| 89 | { | ||
| 90 | StrIntHash h; | ||
| 91 | h.insert("hello", 55 ); | ||
| 92 | h.insert("goodbye", -1812 ); | ||
| 93 | |||
| 94 | StrIntHash h2 = h; | ||
| 95 | unitTest( h2.isCoreShared( h ) ); | ||
| 96 | |||
| 97 | StrIntHash h3 = h.clone(); | ||
| 98 | unitTest( !h3.isCoreShared( h ) ); | ||
| 99 | } | ||
| 87 | } | 100 | } | 
| diff --git a/src/utfstring.cpp b/src/utfstring.cpp new file mode 100644 index 0000000..ae5efaf --- /dev/null +++ b/src/utfstring.cpp | |||
| @@ -0,0 +1,29 @@ | |||
| 1 | #include "bu/utfstring.h" | ||
| 2 | |||
| 3 | template class Bu::FBasicString<short>; | ||
| 4 | |||
| 5 | template<> uint32_t Bu::__calcHashCode<Bu::UtfString>( const Bu::UtfString &k ) | ||
| 6 | { | ||
| 7 | long j, sz = k.getSize()*2; | ||
| 8 | const char *s = (const char *)k.getStr(); | ||
| 9 | |||
| 10 | long nPos = 0; | ||
| 11 | for( j = 0; j < sz; j++, s++ ) | ||
| 12 | { | ||
| 13 | nPos = *s + (nPos << 6) + (nPos << 16) - nPos; | ||
| 14 | } | ||
| 15 | |||
| 16 | return nPos; | ||
| 17 | } | ||
| 18 | |||
| 19 | template<> bool Bu::__cmpHashKeys<Bu::UtfString>( | ||
| 20 | const Bu::UtfString &a, const Bu::UtfString &b ) | ||
| 21 | { | ||
| 22 | return a == b; | ||
| 23 | } | ||
| 24 | |||
| 25 | template<> void Bu::__tracer_format<Bu::UtfString>( const Bu::UtfString &v ) | ||
| 26 | { | ||
| 27 | printf("(%ld)\"%s\"", v.getSize(), (const char *)v.getStr() ); | ||
| 28 | } | ||
| 29 | |||
| diff --git a/src/utfstring.h b/src/utfstring.h new file mode 100644 index 0000000..bbacb74 --- /dev/null +++ b/src/utfstring.h | |||
| @@ -0,0 +1,24 @@ | |||
| 1 | #ifndef BU_UTF_STRING_H | ||
| 2 | #define BU_UTF_STRING_H | ||
| 3 | |||
| 4 | #include "bu/fbasicstring.h" | ||
| 5 | |||
| 6 | namespace Bu | ||
| 7 | { | ||
| 8 | typedef FBasicString<short> UtfString; | ||
| 9 | |||
| 10 | template<typename T> | ||
| 11 | uint32_t __calcHashCode( const T &k ); | ||
| 12 | |||
| 13 | template<typename T> | ||
| 14 | bool __cmpHashKeys( const T &a, const T &b ); | ||
| 15 | |||
| 16 | template<> uint32_t __calcHashCode<UtfString>( const UtfString &k ); | ||
| 17 | template<> bool __cmpHashKeys<UtfString>( | ||
| 18 | const UtfString &a, const UtfString &b ); | ||
| 19 | |||
| 20 | template<typename t> void __tracer_format( const t &v ); | ||
| 21 | template<> void __tracer_format<UtfString>( const UtfString &v ); | ||
| 22 | } | ||
| 23 | |||
| 24 | #endif | ||
| diff --git a/src/variant.h b/src/variant.h index 8e1203c..f659ad2 100644 --- a/src/variant.h +++ b/src/variant.h | |||
| @@ -17,6 +17,7 @@ namespace Bu | |||
| 17 | { | 17 | { | 
| 18 | class Formatter; | 18 | class Formatter; | 
| 19 | class Variant; | 19 | class Variant; | 
| 20 | /** @cond DEVEL */ | ||
| 20 | template<class t> class VariantType; | 21 | template<class t> class VariantType; | 
| 21 | 22 | ||
| 22 | class VariantTypeRoot | 23 | class VariantTypeRoot | 
| @@ -92,7 +93,22 @@ namespace Bu | |||
| 92 | private: | 93 | private: | 
| 93 | t data; | 94 | t data; | 
| 94 | }; | 95 | }; | 
| 95 | 96 | /** @endcond */ | |
| 97 | |||
| 98 | /** | ||
| 99 | * Store any data type and access it safely. Variant gives you a way to | ||
| 100 | * pass arbitrary data types around without having to worry about what | ||
| 101 | * type a variable is. It allows code to be easily extended and to manage | ||
| 102 | * data without having to know what type it is ahead of time. | ||
| 103 | * | ||
| 104 | * Because of the generic method that this class was implemented it may seem | ||
| 105 | * to have some drawbacks compared to other Variant classes you may have | ||
| 106 | * seen, however it is fairly easy to get it to do just about anything you | ||
| 107 | * may need. It is also very low overhead. On most compilers the class | ||
| 108 | * itself has only 3 words of overhead + the size of the variable you store | ||
| 109 | * in it. And, since many parts of it are templatized they can often be | ||
| 110 | * optimized quite a bit. | ||
| 111 | */ | ||
| 96 | class Variant | 112 | class Variant | 
| 97 | { | 113 | { | 
| 98 | public: | 114 | public: | 
