| 
									
										
										
										
											2023-02-18 03:34:40 +00:00
										 |  |  | export module core.sequence; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import core; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export namespace core { | 
					
						
							| 
									
										
										
										
											2023-02-18 19:40:12 +00:00
										 |  |  | 	/**
 | 
					
						
							|  |  |  | 	 * Result codes used by [sequence]-derived types when they are appended to in any way. | 
					
						
							|  |  |  | 	 * | 
					
						
							|  |  |  | 	 * [append_result::ok] indicates that an append operation was successful. | 
					
						
							|  |  |  | 	 * | 
					
						
							|  |  |  | 	 * [append_result::out_of_memory] alerts that the memory required to perform the append | 
					
						
							|  |  |  | 	 * operation failed. | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	enum class [[nodiscard]] append_result { | 
					
						
							|  |  |  | 		ok, | 
					
						
							| 
									
										
										
										
											2023-02-18 03:34:40 +00:00
										 |  |  | 		out_of_memory, | 
					
						
							|  |  |  | 	}; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-18 19:40:12 +00:00
										 |  |  | 	/**
 | 
					
						
							|  |  |  | 	 * Base type for all sequence-like types. | 
					
						
							|  |  |  | 	 * | 
					
						
							|  |  |  | 	 * Sequences are any data structure which owns a linear, non-unique set of elements which may | 
					
						
							|  |  |  | 	 * be queried and/or mutated. | 
					
						
							|  |  |  | 	 */ | 
					
						
							| 
									
										
										
										
											2023-02-18 03:34:40 +00:00
										 |  |  | 	template<typename element> struct sequence { | 
					
						
							|  |  |  | 		virtual ~sequence() {}; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-18 19:40:12 +00:00
										 |  |  | 		/**
 | 
					
						
							|  |  |  | 		 * Attempts to append `source_elements` to the sequence. | 
					
						
							|  |  |  | 		 * | 
					
						
							|  |  |  | 		 * The returned [append_result] indicates whether the operation was successful or not. | 
					
						
							|  |  |  | 		 * | 
					
						
							|  |  |  | 		 * If the returned [append_result] is anything but [append_result::ok], the [sequence] will | 
					
						
							|  |  |  | 		 * be left in an implementation-defined state. | 
					
						
							|  |  |  | 		 */ | 
					
						
							|  |  |  | 		virtual append_result append(slice<element const> const & source_elements) = 0; | 
					
						
							| 
									
										
										
										
											2023-02-18 03:34:40 +00:00
										 |  |  | 	}; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-18 19:40:12 +00:00
										 |  |  | 	/**
 | 
					
						
							|  |  |  | 	 * Last-in-first-out linear sequence of `element` values. | 
					
						
							|  |  |  | 	 * | 
					
						
							|  |  |  | 	 * [stack] types will default to using an inline array of `init_capacity` at first. After all | 
					
						
							|  |  |  | 	 * local storage has been exhausted, the [stack] will switch to a dynamic buffer. Because of | 
					
						
							|  |  |  | 	 * this, it is recommended to use larger `init_capacity` values for data which has a known or | 
					
						
							|  |  |  | 	 * approximate upper bound at compile-time. Otherwise, the `init_capacity` value may be left at | 
					
						
							|  |  |  | 	 * its default. | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	template<typename element, usize init_capacity = 1> struct stack : public sequence<element> { | 
					
						
							|  |  |  | 		stack(allocator * dynamic_allocator) : local_buffer{0} { | 
					
						
							|  |  |  | 			this->dynamic_allocator = dynamic_allocator; | 
					
						
							|  |  |  | 			this->filled = 0; | 
					
						
							|  |  |  | 			this->elements = this->local_buffer; | 
					
						
							| 
									
										
										
										
											2023-02-18 03:34:40 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		~stack() override { | 
					
						
							| 
									
										
										
										
											2023-02-18 19:40:12 +00:00
										 |  |  | 			if (this->is_dynamic()) this->dynamic_allocator->deallocate(this->elements.pointer); | 
					
						
							| 
									
										
										
										
											2023-02-18 03:34:40 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-18 19:40:12 +00:00
										 |  |  | 		/**
 | 
					
						
							|  |  |  | 		 * Attempts to append `source_elements` to the top of the stack. | 
					
						
							|  |  |  | 		 * | 
					
						
							|  |  |  | 		 * The returned [append_result] indicates whether the operation was successful or not. | 
					
						
							|  |  |  | 		 * | 
					
						
							|  |  |  | 		 * If the returned [append_result] is anything but [append_result::ok], the stack will be | 
					
						
							|  |  |  | 		 * be left in an empty but valid state. | 
					
						
							|  |  |  | 		 * | 
					
						
							|  |  |  | 		 * *Note* that [push] is recommended when appending singular values. | 
					
						
							|  |  |  | 		 */ | 
					
						
							|  |  |  | 		append_result append(slice<element const> const & source_elements) override { | 
					
						
							|  |  |  | 			usize const updated_fill = this->filled + source_elements.length; | 
					
						
							| 
									
										
										
										
											2023-02-18 03:34:40 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-18 19:40:12 +00:00
										 |  |  | 			if (updated_fill >= this->elements.length) { | 
					
						
							|  |  |  | 				append_result const result = | 
					
						
							|  |  |  | 					this->reserve(max(this->elements.length, updated_fill)); | 
					
						
							| 
									
										
										
										
											2023-02-18 03:34:40 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-18 19:40:12 +00:00
										 |  |  | 				if (result != append_result::ok) return result; | 
					
						
							| 
									
										
										
										
											2023-02-18 03:34:40 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-18 19:40:12 +00:00
										 |  |  | 			for (usize i = 0; i < source_elements.length; i += 1) | 
					
						
							|  |  |  | 				this->elements[this->filled + i] = source_elements[i]; | 
					
						
							| 
									
										
										
										
											2023-02-18 03:34:40 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-18 19:40:12 +00:00
										 |  |  | 			this->filled = updated_fill; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			return append_result::ok; | 
					
						
							| 
									
										
										
										
											2023-02-18 03:34:40 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-18 19:40:12 +00:00
										 |  |  | 		/**
 | 
					
						
							|  |  |  | 		 * Returns the beginning of the elements as a mutable pointer. | 
					
						
							|  |  |  | 		 */ | 
					
						
							|  |  |  | 		element * begin() { | 
					
						
							|  |  |  | 			return this->elements.pointer; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2023-02-18 03:34:40 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-18 19:40:12 +00:00
										 |  |  | 		/**
 | 
					
						
							|  |  |  | 		 * Returns the beginning of the elements as a const pointer. | 
					
						
							|  |  |  | 		 */ | 
					
						
							|  |  |  | 		element const * begin() const { | 
					
						
							|  |  |  | 			return this->elements.pointer; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2023-02-18 03:34:40 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-18 19:40:12 +00:00
										 |  |  | 		/**
 | 
					
						
							|  |  |  | 		 * Returns the ending of the elements as a mutable pointer. | 
					
						
							|  |  |  | 		 */ | 
					
						
							|  |  |  | 		element * end() { | 
					
						
							|  |  |  | 			return this->elements.pointer + this->filled; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/**
 | 
					
						
							|  |  |  | 		 * Returns the ending of the elements as a const pointer. | 
					
						
							|  |  |  | 		 */ | 
					
						
							|  |  |  | 		element const * end() const { | 
					
						
							|  |  |  | 			return this->elements.pointer + this->filled; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2023-02-18 03:34:40 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-18 19:40:12 +00:00
										 |  |  | 		/**
 | 
					
						
							|  |  |  | 		 * Returns `true` if the stack is backed by dynamic memory, otherwise `false`. | 
					
						
							|  |  |  | 		 */ | 
					
						
							|  |  |  | 		bool is_dynamic() const { | 
					
						
							|  |  |  | 			return this->elements.pointer != this->local_buffer; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2023-02-18 03:34:40 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-18 19:40:12 +00:00
										 |  |  | 		/**
 | 
					
						
							|  |  |  | 		 * Attempts to append `source_element` to the top of the stack. | 
					
						
							|  |  |  | 		 * | 
					
						
							|  |  |  | 		 * The returned [append_result] indicates whether the operation was successful or not. | 
					
						
							|  |  |  | 		 * | 
					
						
							|  |  |  | 		 * If the returned [append_result] is anything but [append_result::ok], the stack will be | 
					
						
							|  |  |  | 		 * be left in an empty but valid state. | 
					
						
							|  |  |  | 		 * | 
					
						
							|  |  |  | 		 * *Note* that [append] is recommended when appending many values at once. | 
					
						
							|  |  |  | 		 */ | 
					
						
							|  |  |  | 		append_result push(element const & source_element) { | 
					
						
							|  |  |  | 			if (this->filled == this->elements.length) { | 
					
						
							|  |  |  | 				append_result const result = this->reserve(this->elements.length); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				if (result != append_result::ok) return result; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			this->elements[this->filled] = source_element; | 
					
						
							|  |  |  | 			this->filled += 1; | 
					
						
							| 
									
										
										
										
											2023-02-18 03:34:40 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-18 19:40:12 +00:00
										 |  |  | 			return append_result::ok; | 
					
						
							| 
									
										
										
										
											2023-02-18 03:34:40 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-18 19:40:12 +00:00
										 |  |  | 		/**
 | 
					
						
							|  |  |  | 		 * Attempts to reserve `capacity` number of elements additional space on the stack, forcing | 
					
						
							|  |  |  | 		 * it to use dynamic memory _even_ if it hasn't exhausted the local buffer yet. | 
					
						
							|  |  |  | 		 * | 
					
						
							|  |  |  | 		 * The returned [append_result] indicates whether the operation was successful or not. | 
					
						
							|  |  |  | 		 * | 
					
						
							|  |  |  | 		 * If the returned [append_result] is anything but [append_result::ok], the stack will be | 
					
						
							|  |  |  | 		 * be left in an empty but valid state. | 
					
						
							|  |  |  | 		 * | 
					
						
							|  |  |  | 		 * *Note* that manual invocation is not recommended if the [stack] has a large | 
					
						
							|  |  |  | 		 * `initial_capacity` argument. | 
					
						
							|  |  |  | 		 */ | 
					
						
							|  |  |  | 		append_result reserve(usize capacity) { | 
					
						
							| 
									
										
										
										
											2023-02-18 03:34:40 +00:00
										 |  |  | 			usize const requested_capacity = this->elements.length + capacity; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-18 19:40:12 +00:00
										 |  |  | 			if (this->is_dynamic()) { | 
					
						
							|  |  |  | 				// Grow dynamic buffer (bailing out if failed).
 | 
					
						
							|  |  |  | 				u8 * const buffer = this->dynamic_allocator->reallocate( | 
					
						
							|  |  |  | 					reinterpret_cast<u8 *>(this->elements.pointer), | 
					
						
							|  |  |  | 						sizeof(element) * requested_capacity); | 
					
						
							| 
									
										
										
										
											2023-02-18 03:34:40 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-18 19:40:12 +00:00
										 |  |  | 				if (buffer == nullptr) { | 
					
						
							| 
									
										
										
										
											2023-02-18 03:34:40 +00:00
										 |  |  | 					this->elements = {}; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-18 19:40:12 +00:00
										 |  |  | 					return append_result::out_of_memory; | 
					
						
							| 
									
										
										
										
											2023-02-18 03:34:40 +00:00
										 |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-18 19:40:12 +00:00
										 |  |  | 				this->elements = {reinterpret_cast<element *>(buffer), requested_capacity}; | 
					
						
							| 
									
										
										
										
											2023-02-18 03:34:40 +00:00
										 |  |  | 			} else { | 
					
						
							| 
									
										
										
										
											2023-02-18 19:40:12 +00:00
										 |  |  | 				usize const buffer_size = sizeof(element) * requested_capacity; | 
					
						
							|  |  |  | 				u8 * const buffer = this->dynamic_allocator->reallocate(nullptr, buffer_size); | 
					
						
							| 
									
										
										
										
											2023-02-18 03:34:40 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-18 19:40:12 +00:00
										 |  |  | 				if (buffer == nullptr) { | 
					
						
							| 
									
										
										
										
											2023-02-18 03:34:40 +00:00
										 |  |  | 					this->elements = {}; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-18 19:40:12 +00:00
										 |  |  | 					return append_result::out_of_memory; | 
					
						
							| 
									
										
										
										
											2023-02-18 03:34:40 +00:00
										 |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-18 19:40:12 +00:00
										 |  |  | 				core::copy({buffer, buffer_size}, this->elements.as_bytes()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				this->elements = {reinterpret_cast<element *>(buffer), requested_capacity}; | 
					
						
							| 
									
										
										
										
											2023-02-18 03:34:40 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-18 19:40:12 +00:00
										 |  |  | 			return append_result::ok; | 
					
						
							| 
									
										
										
										
											2023-02-18 03:34:40 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		private: | 
					
						
							| 
									
										
										
										
											2023-02-18 19:40:12 +00:00
										 |  |  | 		allocator * dynamic_allocator; | 
					
						
							| 
									
										
										
										
											2023-02-18 03:34:40 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		usize filled; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		slice<element> elements; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-18 19:40:12 +00:00
										 |  |  | 		element local_buffer[init_capacity]; | 
					
						
							| 
									
										
										
										
											2023-02-18 03:34:40 +00:00
										 |  |  | 	}; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-18 19:40:12 +00:00
										 |  |  | 	/**
 | 
					
						
							|  |  |  | 	 * Writable type for appending data to a [sequence] containing [u8] values. | 
					
						
							|  |  |  | 	 */ | 
					
						
							| 
									
										
										
										
											2023-02-18 03:34:40 +00:00
										 |  |  | 	struct sequence_writer : public writable { | 
					
						
							| 
									
										
										
										
											2023-02-18 19:40:12 +00:00
										 |  |  | 		sequence_writer(sequence<u8> * output_sequence) : writable{ | 
					
						
							|  |  |  | 			[output_sequence](slice<u8 const> const & buffer) -> expected<usize, io_error> { | 
					
						
							|  |  |  | 				switch (output_sequence->append(buffer)) { | 
					
						
							|  |  |  | 					case append_result::ok: return buffer.length; | 
					
						
							|  |  |  | 					case append_result::out_of_memory: return io_error::unavailable; | 
					
						
							|  |  |  | 					default: unreachable(); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			}} {} | 
					
						
							| 
									
										
										
										
											2023-02-18 03:34:40 +00:00
										 |  |  | 	}; | 
					
						
							|  |  |  | } |