| /* |
| * Licensed to the Apache Software Foundation (ASF) under one or more |
| * contributor license agreements. See the NOTICE file distributed with |
| * this work for additional information regarding copyright ownership. |
| * The ASF licenses this file to You under the Apache License, Version 2.0 |
| * (the "License"); you may not use this file except in compliance with |
| * the License. You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| |
| #ifndef _DECAF_NIO_CHARBUFFER_H_ |
| #define _DECAF_NIO_CHARBUFFER_H_ |
| |
| #include <decaf/nio/Buffer.h> |
| #include <decaf/lang/Comparable.h> |
| #include <decaf/lang/exceptions/NullPointerException.h> |
| #include <decaf/lang/exceptions/IndexOutOfBoundsException.h> |
| #include <decaf/nio/BufferUnderflowException.h> |
| #include <decaf/nio/BufferOverflowException.h> |
| #include <decaf/nio/ReadOnlyBufferException.h> |
| #include <decaf/lang/CharSequence.h> |
| #include <decaf/lang/Appendable.h> |
| |
| namespace decaf{ |
| namespace nio{ |
| |
| /** |
| * This class defines four categories of operations upon character buffers: |
| * |
| * o Absolute and relative get and put methods that read and write single characters; |
| * o Relative bulk get methods that transfer contiguous sequences of characters from |
| * this buffer into an array; and |
| * o Relative bulk put methods that transfer contiguous sequences of characters from |
| * a character array, a string, or some other character buffer into this buffer. |
| * o Methods for compacting, duplicating, and slicing a character buffer. |
| * |
| * Character buffers can be created either by allocation, which allocates space for |
| * the buffer's content, by wrapping an existing character array or string into a |
| * buffer, or by creating a view of an existing byte buffer |
| * |
| * This class implements the CharSequence interface so that character buffers may |
| * be used wherever character sequences are accepted, for example in the |
| * regular-expression package decaf.util.regex. |
| * |
| * Methods in this class that do not otherwise have a value to return are specified |
| * to return the buffer upon which they are invoked. This allows method invocations |
| * to be chained. The sequence of statements |
| * |
| * cb.put("text/"); |
| * cb.put(subtype); |
| * cb.put("; charset="); |
| * cb.put(enc); |
| * |
| * can, for example, be replaced by the single statement |
| * |
| * cb.put("text/").put(subtype).put("; charset=").put(enc); |
| */ |
| class DECAF_API CharBuffer : public Buffer, |
| public lang::CharSequence, |
| public lang::Appendable, |
| public lang::Comparable<CharBuffer> { |
| protected: |
| |
| /** |
| * Creates a CharBuffer object that has its backing array allocated internally |
| * and is then owned and deleted when this object is deleted. The array is |
| * initially created with all elements initialized to zero. |
| * |
| * @param capacity |
| * The size of the array, this is the limit we read and write to. |
| * |
| * @throws IllegalArguementException if capacity is negative. |
| */ |
| CharBuffer( int capacity ); |
| |
| public: |
| |
| virtual ~CharBuffer() {} |
| |
| /** |
| * @return a std::string describing this object |
| */ |
| virtual std::string toString() const; |
| |
| /** |
| * Appends the specified character to this buffer. |
| * |
| * @param value |
| * The char to append. |
| * |
| * @return a reference to this modified CharBuffer. |
| * |
| * @throws BufferOverflowException if there is no more space |
| * @throws ReadOnlyBufferException if this Buffer is read only. |
| */ |
| CharBuffer& append( char value ); |
| |
| /** |
| * Appends the specified character sequence to this buffer. |
| * If value is Null the the string "null" is appended to the buffer. |
| * |
| * @param value |
| * The CharSequence to append. |
| * |
| * @return a reference to this modified CharBuffer |
| * |
| * @throws BufferOverflowException if there is no more space |
| * @throws ReadOnlyBufferException if this Buffer is read only. |
| */ |
| CharBuffer& append( const lang::CharSequence* value ); |
| |
| /** |
| * Appends a subsequence of the specified character sequence to this buffer |
| * If value is Null the the string "null" is appended to the buffer. |
| * |
| * @param value |
| * The CharSequence to append. |
| * @param start |
| * The index to start appending from. |
| * @param end |
| * The index to append to. |
| * |
| * @return a reference to this modified CharBuffer. |
| * |
| * @throws BufferOverflowException if there is no more space |
| * @throws ReadOnlyBufferException if this Buffer is read only. |
| * @throws IndexOutOfBoundsException if start > end, or > length of sequence. |
| */ |
| CharBuffer& append( const lang::CharSequence* value, int start, int end ); |
| |
| /** |
| * Returns the character array that backs this buffer (optional operation). |
| * |
| * Modifications to this buffer's content will cause the returned array's content |
| * to be modified, and vice versa. |
| * |
| * Invoke the hasArray method before invoking this method in order to ensure that |
| * this buffer has an accessible backing array. |
| * |
| * @return the array that backs this Buffer. |
| * |
| * @throws ReadOnlyBufferException if this Buffer is read only. |
| * @throws UnsupportedOperationException if the underlying store has no array. |
| */ |
| virtual char* array() = 0; |
| |
| /** |
| * Returns the offset within this buffer's backing array of the first element of |
| * the buffer (optional operation). |
| * |
| * Invoke the hasArray method before invoking this method in order to ensure that |
| * this buffer has an accessible backing array. |
| * |
| * @return The offset into the backing array where index zero starts. |
| * |
| * @throws ReadOnlyBufferException if this Buffer is read only. |
| * @throws UnsupportedOperationException if the underlying store has no array. |
| */ |
| virtual int arrayOffset() = 0; |
| |
| /** |
| * Creates a new, read-only char buffer that shares this buffer's content. |
| * |
| * The content of the new buffer will be that of this buffer. Changes to this |
| * buffer's content will be visible in the new buffer; the new buffer itself, |
| * however, will be read-only and will not allow the shared content to be |
| * modified. The two buffers' position, limit, and mark values will be |
| * independent. |
| * |
| * If this buffer is itself read-only then this method behaves in exactly the |
| * same way as the duplicate method. |
| * |
| * The new buffer's capacity, limit, position, and mark values will be |
| * identical to those of this buffer. |
| * |
| * @return The new, read-only char buffer which the caller then owns. |
| */ |
| virtual CharBuffer* asReadOnlyBuffer() const = 0; |
| |
| /** |
| * Reads the character at the given index relative to the current position. |
| * |
| * @param index - The index of the character to be read relative to position |
| * |
| * @return The character at index position() + index. |
| * |
| * @throws IndexOutOfBoundsException if the index + the current position exceeds the |
| * size of the buffer or the index is negative. |
| */ |
| char charAt( int index ) const; |
| |
| /** |
| * Compacts this buffer |
| * |
| * The bytes between the buffer's current position and its limit, if any, are |
| * copied to the beginning of the buffer. That is, the byte at index |
| * p = position() is copied to index zero, the byte at index p + 1 is copied |
| * to index one, and so forth until the byte at index limit() - 1 is copied |
| * to index n = limit() - 1 - p. The buffer's position is then set to n+1 and |
| * its limit is set to its capacity. The mark, if defined, is discarded. |
| * |
| * The buffer's position is set to the number of bytes copied, rather than to |
| * zero, so that an invocation of this method can be followed immediately by |
| * an invocation of another relative put method. |
| * |
| * @return a reference to this CharBuffer. |
| * |
| * @throws ReadOnlyBufferException - If this buffer is read-only |
| */ |
| virtual CharBuffer& compact() = 0; |
| |
| /** |
| * Creates a new char buffer that shares this buffer's content. |
| * |
| * The content of the new buffer will be that of this buffer. Changes to this |
| * buffer's content will be visible in the new buffer, and vice versa; the two |
| * buffers' position, limit, and mark values will be independent. |
| * |
| * The new buffer's capacity, limit, position, and mark values will be identical |
| * to those of this buffer. The new buffer will be read-only if, and only if, |
| * this buffer is read-only. |
| * |
| * @return a new char Buffer which the caller owns. |
| */ |
| virtual CharBuffer* duplicate() = 0; |
| |
| /** |
| * Relative get method. Reads the character at this buffer's current position, |
| * and then increments the position. |
| * |
| * @return the char at the current position. |
| * |
| * @throws BufferUnderflowException if there no more data to return |
| */ |
| virtual char get() = 0; |
| |
| /** |
| * Absolute get method. Reads the char at the given index. |
| * |
| * @param index |
| * The index in the Buffer where the char is to be read. |
| * |
| * @return the char that is located at the given index. |
| * |
| * @throws IndexOutOfBoundsException if index is not smaller than the |
| * buffer's limit or is negative. |
| */ |
| virtual char get( int index ) const = 0; |
| |
| /** |
| * Relative bulk get method. |
| * |
| * This method transfers chars from this buffer into the given destination |
| * vector. An invocation of this method of the form src.get(a) behaves in |
| * exactly the same way as the invocation. The vector must be sized to the |
| * amount of data that is to be read, that is to say, the caller should call |
| * buffer.resize( N ) before calling this get method. |
| * |
| * @return a reference to this CharBuffer. |
| * |
| * @throws BufferUnderflowException if there are fewer than length chars |
| * remaining in this buffer. |
| */ |
| CharBuffer& get( std::vector<char> buffer ); |
| |
| /** |
| * Relative bulk get method. |
| * |
| * This method transfers chars from this buffer into the given destination array. |
| * If there are fewer chars remaining in the buffer than are required to satisfy |
| * the request, that is, if length > remaining(), then no bytes are transferred |
| * and a BufferUnderflowException is thrown. |
| * |
| * Otherwise, this method copies length chars from this buffer into the given |
| * array, starting at the current position of this buffer and at the given offset |
| * in the array. The position of this buffer is then incremented by length. |
| * |
| * @param buffer |
| * The pointer to an allocated buffer to fill. |
| * @param size |
| * The size of the buffer passed. |
| * @param offset |
| * The position in the buffer to start filling. |
| * @param length |
| * The amount of data to put in the passed buffer. |
| * |
| * @return a reference to this Buffer. |
| * |
| * @throws BufferUnderflowException if there are fewer than length chars |
| * remaining in this buffer |
| * @throws NullPointerException if the passed buffer is null. |
| * @throws IndexOutOfBoundsException if the preconditions of size, offset, or length |
| * are not met. |
| */ |
| CharBuffer& get( char* buffer, int size, int offset, int length ); |
| |
| /** |
| * Tells whether or not this buffer is backed by an accessible char array. |
| * If this method returns true then the array and arrayOffset methods may safely |
| * be invoked. Subclasses should override this method if they do not have a |
| * backing array as this class always returns true. |
| * |
| * @return true if, and only if, this buffer is backed by an array and is not |
| * read-only |
| */ |
| virtual bool hasArray() const = 0; |
| |
| /** |
| * Returns the length of this character buffer. |
| * |
| * @return the length of this buffer from the position to the limit. |
| */ |
| int length() const { |
| return this->remaining(); |
| } |
| |
| /** |
| * This method transfers the chars remaining in the given source buffer into |
| * this buffer. If there are more chars remaining in the source buffer than in |
| * this buffer, that is, if src.remaining() > remaining(), then no chars are |
| * transferred and a BufferOverflowException is thrown. |
| * |
| * Otherwise, this method copies n = src.remaining() chars from the given |
| * buffer into this buffer, starting at each buffer's current position. The |
| * positions of both buffers are then incremented by n. |
| * |
| * @param src - the buffer to take chars from an place in this one. |
| * |
| * @return a reference to this buffer. |
| * |
| * @throws BufferOverflowException if there is insufficient space in this buffer |
| * for the remaining chars in the source buffer. |
| * @throws IllegalArgumentException if the source buffer is this buffer. |
| * @throws ReadOnlyBufferException if this buffer is read-only. |
| */ |
| CharBuffer& put( CharBuffer& src ); |
| |
| /** |
| * This method transfers chars into this buffer from the given source array. |
| * If there are more chars to be copied from the array than remain in this buffer, |
| * that is, if length > remaining(), then no chars are transferred and a |
| * BufferOverflowException is thrown. |
| * |
| * Otherwise, this method copies length bytes from the given array into this |
| * buffer, starting at the given offset in the array and at the current position |
| * of this buffer. The position of this buffer is then incremented by length. |
| * |
| * @param buffer |
| * The array from which chars are to be read. |
| * @param size |
| * The size of the buffer passed. |
| * @param offset |
| * The offset within the array of the first char to be read. |
| * @param length |
| * The number of chars to be read from the given array. |
| * |
| * @return a reference to this buffer. |
| * |
| * @throws BufferOverflowException if there is insufficient space in this buffer |
| * @throws ReadOnlyBufferException if this buffer is read-only |
| * @throws NullPointerException if the passed buffer is null. |
| * @throws IndexOutOfBoundsException if the preconditions of size, offset, or length |
| * are not met. |
| */ |
| CharBuffer& put( const char* buffer, int size, int offset, int length ); |
| |
| /** |
| * This method transfers the entire content of the given source char array into |
| * this buffer. This is the same as calling put( &buffer[0], 0, buffer.size(). |
| * |
| * @param buffer |
| * The buffer whose contents are copied to this CharBuffer. |
| * |
| * @return a reference to this buffer. |
| * |
| * @throws BufferOverflowException if there is insufficient space in this buffer. |
| * @throws ReadOnlyBufferException if this buffer is read-only. |
| */ |
| CharBuffer& put( std::vector<char>& buffer ); |
| |
| /** |
| * Writes the given char into this buffer at the current position, and then |
| * increments the position. |
| * |
| * @param value |
| * The char value to be written. |
| * |
| * @return a reference to this buffer. |
| * |
| * @throws BufferOverflowException if this buffer's current position is not |
| * smaller than its limit |
| * @throws ReadOnlyBufferException if this buffer is read-only. |
| */ |
| virtual CharBuffer& put( char value ) = 0; |
| |
| /** |
| * Writes the given char into this buffer at the given index. |
| * |
| * @param index |
| * The position in the Buffer to write the data. |
| * @param value |
| * The char to write. |
| * |
| * @return a reference to this buffer. |
| * |
| * @throws IndexOutOfBoundsException if index greater than the buffer's limit |
| * minus the size of the type being written, or index is negative. |
| * @throws ReadOnlyBufferException if this buffer is read-only. |
| */ |
| virtual CharBuffer& put( int index, char value ) = 0; |
| |
| /** |
| * Relative bulk put method (optional operation). |
| * |
| * This method transfers characters from the given string into this buffer. If |
| * there are more characters to be copied from the string than remain in this |
| * buffer, that is, if end - start > remaining(), then no characters are |
| * transferred and a BufferOverflowException is thrown. |
| * @return a reference to this buffer |
| * |
| * Otherwise, this method copies n = end - start characters from the given string |
| * into this buffer, starting at the given start index and at the current position |
| * of this buffer. The position of this buffer is then incremented by n. |
| * |
| * @param src |
| * The string to copy from. |
| * @param start |
| * The position in src to start from. |
| * @param end |
| * The position in src to stop at. |
| * |
| * @return a reference to this CharBuffer. |
| * |
| * @throws BufferOverflowException if this buffer's current position is not |
| * @throws IndexOutOfBoundsException if index greater than the buffer's limit |
| * minus the size of the type being written. |
| * @throws ReadOnlyBufferException if this buffer is read-only |
| */ |
| CharBuffer& put( std::string& src, int start, int end ); |
| |
| /** |
| * Relative bulk put method (optional operation). |
| * |
| * This method transfers the entire content of the given source string into this |
| * buffer. An invocation of this method of the form dst.put(s) behaves in exactly |
| * the same way as the invocation. |
| * |
| * @param src |
| * The string to copy from. |
| * |
| * @return a reference to this CharBuffer. |
| * |
| * @throws BufferOverflowException if this buffer's current position is not. |
| * @throws ReadOnlyBufferException if this buffer is read-only. |
| */ |
| CharBuffer& put( const std::string& src ); |
| |
| /** |
| * Attempts to read characters into the specified character buffer. The buffer is |
| * used as a repository of characters as-is: the only changes made are the results |
| * of a put operation. No flipping or rewinding of the buffer is performed. |
| * |
| * @param target |
| * The buffer to read characters into |
| * |
| * @return The number of characters added to the buffer, or string::npos if this |
| * source of characters is at its end |
| * |
| * @throws NullPointerException if target is Null. |
| * @throws IllegalArgumentException if target is this CharBuffer. |
| * @throws ReadOnlyBufferException if this buffer is in read-only mode. |
| */ |
| virtual int read( CharBuffer* target ); |
| |
| /** |
| * Creates a new character buffer that represents the specified subsequence of |
| * this buffer, relative to the current position. |
| * |
| * The new buffer will share this buffer's content; that is, if the content of |
| * this buffer is mutable then modifications to one buffer will cause the other |
| * to be modified. The new buffer's capacity will be that of this buffer, its |
| * position will be position() + start, and its limit will be position() + end. |
| * The new Buffer will be read-only if, and only if, this buffer is read-only. |
| * |
| * @param start |
| * The index, relative to the current position, of the first character in |
| * the subsequence; must be non-negative and no larger than remaining(). |
| * @param end |
| * The index, relative to the current position, of the character following the |
| * last character in the subsequence; must be no smaller than start and no |
| * larger than remaining(). |
| * |
| * @return The new character buffer, caller owns. |
| * |
| * @throws IndexOutOfBoundsException if the preconditions on start and end fail. |
| */ |
| virtual lang::CharSequence* subSequence( int start, int end ) const = 0; |
| |
| /** |
| * Creates a new CharBuffer whose content is a shared subsequence of this |
| * buffer's content. The content of the new buffer will start at this buffer's |
| * current position. Changes to this buffer's content will be visible in the new |
| * buffer, and vice versa; the two buffers' position, limit, and mark values will |
| * be independent. |
| * |
| * The new buffer's position will be zero, its capacity and its limit will be the |
| * number of bytes remaining in this buffer, and its mark will be undefined. The |
| * new buffer will be read-only if, and only if, this buffer is read-only. |
| * |
| * @return the newly create CharBuffer which the caller owns. |
| */ |
| virtual CharBuffer* slice() const = 0; |
| |
| public: // Comparable |
| |
| /** |
| * {@inheritDoc} |
| */ |
| virtual int compareTo( const CharBuffer& value ) const; |
| |
| /** |
| * {@inheritDoc} |
| */ |
| virtual bool equals( const CharBuffer& value ) const; |
| |
| /** |
| * {@inheritDoc} |
| */ |
| virtual bool operator==( const CharBuffer& value ) const; |
| |
| /** |
| * {@inheritDoc} |
| */ |
| virtual bool operator<( const CharBuffer& value ) const; |
| |
| public: // Statics |
| |
| /** |
| * Allocates a new character buffer. |
| * |
| * The new buffer's position will be zero, its limit will be its capacity, and |
| * its mark will be undefined. It will have a backing array, and its array offset |
| * will be zero. |
| * |
| * @param capacity |
| * The size of the Char buffer in chars ( 1 byte ). |
| * |
| * @return the CharBuffer that was allocated, caller owns. |
| * |
| * @throws IndexOutOfBoundsException if capacity is negative. |
| */ |
| static CharBuffer* allocate( int capacity ); |
| |
| /** |
| * Wraps the passed buffer with a new CharBuffer. |
| * |
| * The new buffer will be backed by the given char array; that is, modifications |
| * to the buffer will cause the array to be modified and vice versa. The new |
| * buffer's capacity will be array.length, its position will be offset, its limit |
| * will be offset + length, and its mark will be undefined. Its backing array |
| * will be the given array, and its array offset will be zero. |
| * |
| * @param array |
| * The array that will back the new buffer. |
| * @param size |
| * The size of the array passed in. |
| * @param offset |
| * The offset of the subarray to be used. |
| * @param length |
| * The length of the subarray to be used. |
| * |
| * @return a new CharBuffer that is backed by buffer, caller owns. |
| * |
| * @throws NullPointerException if the array pointer is Null. |
| * @throws IndexOutOfBoundsException if capacity is negative. |
| */ |
| static CharBuffer* wrap( char* array, int size, int offset, int length ); |
| |
| /** |
| * Wraps the passed STL char Vector in a CharBuffer. |
| * |
| * The new buffer will be backed by the given char array; modifications to the |
| * buffer will cause the array to be modified and vice versa. The new buffer's |
| * capacity and limit will be buffer.size(), its position will be zero, and its |
| * mark will be undefined. Its backing array will be the given array, and its |
| * array offset will be zero. |
| * |
| * @param buffer |
| * The vector that will back the new buffer, the vector must have been |
| * sized to the desired size already by calling vector.resize( N ). |
| * |
| * @return a new CharBuffer that is backed by buffer, caller owns. |
| */ |
| static CharBuffer* wrap( std::vector<char>& buffer ); |
| |
| /** |
| * Wraps a character sequence into a buffer. |
| * |
| * The content of the new, read-only buffer will be the content of the given |
| * character sequence. The buffer's capacity will be csq.length(), its position |
| * will be start, its limit will be end, and its mark will be undefined. |
| * |
| * @param csq |
| * The CharSequence that will back the new buffer |
| * @param start |
| * The index of the first character to be used; must be non-negative and no |
| * larger than csq.length(). The new buffer's position will be set to this value. |
| * @param end |
| * The index of the character following the last character to be used; must be |
| * no smaller than start and no larger than the size of the sequence. |
| * |
| * @return a ReadOnly CharBuffer, caller owns the returned pointer. |
| * |
| * @throws NullPointerException if csq is null. |
| * @throws IndexOutOfBoundsException if the preconditions on start and end fail |
| */ |
| // TODO |
| // static CharBuffer* wrap( lang::CharSequence* csq, int start, int end ); |
| |
| /** |
| * Wraps a full CharSequence into a buffer. |
| * |
| * The content of the new, read-only buffer will be the content of the given |
| * string. The new buffer's capacity and limit will be csq.length(), its position |
| * will be zero, and its mark will be undefined. |
| * |
| * @param csq |
| * The character sequence from which the new character buffer is to |
| * be created, cannot be null. |
| * |
| * @return the newly created CharBuffer, caller owns. |
| * |
| * @throws NullPointerException if csq is null. |
| */ |
| // TODO |
| // static CharBuffer* wrap( lang::CharSequence* csq ); |
| |
| }; |
| |
| }} |
| |
| #endif /*_DECAF_NIO_CHARBUFFER_H_*/ |