Sergey Mashkov | 86f7061 | 2017-07-25 10:59:44 +0300 | [diff] [blame] | 1 | package kotlinx.coroutines.experimental.io.internal |
| 2 | |
Roman Elizarov | 8376203 | 2017-08-08 19:58:40 +0300 | [diff] [blame] | 3 | import java.nio.ByteBuffer |
Sergey Mashkov | 86f7061 | 2017-07-25 10:59:44 +0300 | [diff] [blame] | 4 | |
Roman Elizarov | 8376203 | 2017-08-08 19:58:40 +0300 | [diff] [blame] | 5 | // this is MAGICAL constant that is tied to the code ByteBufferChannel (that is how much it needs extra) |
| 6 | internal const val RESERVED_SIZE = 8 |
Sergey Mashkov | 86f7061 | 2017-07-25 10:59:44 +0300 | [diff] [blame] | 7 | |
Roman Elizarov | 8376203 | 2017-08-08 19:58:40 +0300 | [diff] [blame] | 8 | internal val EmptyByteBuffer: ByteBuffer = ByteBuffer.allocate(0) |
Roman Elizarov | 469cad3 | 2017-08-15 15:54:56 +0300 | [diff] [blame] | 9 | internal val EmptyCapacity = RingBufferCapacity(0) |
Roman Elizarov | 8376203 | 2017-08-08 19:58:40 +0300 | [diff] [blame] | 10 | |
| 11 | internal sealed class ReadWriteBufferState( |
Roman Elizarov | 469cad3 | 2017-08-15 15:54:56 +0300 | [diff] [blame] | 12 | @JvmField val backingBuffer: ByteBuffer, |
| 13 | @JvmField val capacity: RingBufferCapacity |
Roman Elizarov | 8376203 | 2017-08-08 19:58:40 +0300 | [diff] [blame] | 14 | ) { |
Sergey Mashkov | 86f7061 | 2017-07-25 10:59:44 +0300 | [diff] [blame] | 15 | open val idle: Boolean get() = false |
Roman Elizarov | 8376203 | 2017-08-08 19:58:40 +0300 | [diff] [blame] | 16 | open val readBuffer: ByteBuffer get() = error("read buffer is not available in state $this") |
| 17 | open val writeBuffer: ByteBuffer get() = error("write buffer is not available in state $this") |
Sergey Mashkov | 86f7061 | 2017-07-25 10:59:44 +0300 | [diff] [blame] | 18 | |
Roman Elizarov | 8376203 | 2017-08-08 19:58:40 +0300 | [diff] [blame] | 19 | internal open fun startReading(): ReadWriteBufferState = error("Reading is not available in state $this") |
| 20 | internal open fun startWriting(): ReadWriteBufferState = error("Writing is not available in state $this") |
| 21 | internal open fun stopReading(): ReadWriteBufferState = error("Unable to stop reading in state $this") |
| 22 | internal open fun stopWriting(): ReadWriteBufferState = error("Unable to stop writing in state $this") |
Sergey Mashkov | 86f7061 | 2017-07-25 10:59:44 +0300 | [diff] [blame] | 23 | |
Roman Elizarov | 469cad3 | 2017-08-15 15:54:56 +0300 | [diff] [blame] | 24 | object IdleEmpty : ReadWriteBufferState(EmptyByteBuffer, EmptyCapacity) { |
Sergey Mashkov | 86f7061 | 2017-07-25 10:59:44 +0300 | [diff] [blame] | 25 | override val idle: Boolean get() = true |
Sergey Mashkov | 86f7061 | 2017-07-25 10:59:44 +0300 | [diff] [blame] | 26 | override fun toString() = "IDLE(empty)" |
Sergey Mashkov | 86f7061 | 2017-07-25 10:59:44 +0300 | [diff] [blame] | 27 | } |
| 28 | |
Roman Elizarov | 8376203 | 2017-08-08 19:58:40 +0300 | [diff] [blame] | 29 | class Initial( |
| 30 | backingBuffer: ByteBuffer, |
| 31 | reservedSize: Int = RESERVED_SIZE |
| 32 | ) : ReadWriteBufferState(backingBuffer, RingBufferCapacity(backingBuffer.capacity() - reservedSize)) { |
| 33 | init { |
| 34 | require(backingBuffer.position() == 0) |
| 35 | require(backingBuffer.limit() == backingBuffer.capacity()) |
| 36 | } |
| 37 | override val writeBuffer: ByteBuffer = backingBuffer.duplicate() // defensive copy of buffer's state |
| 38 | override val readBuffer: ByteBuffer = backingBuffer.duplicate() // must have a separate buffer state here |
| 39 | // all other possible states |
| 40 | internal val idleState = IdleNonEmpty(this) |
| 41 | internal val readingState = Reading(this) |
| 42 | internal val writingState = Writing(this) |
| 43 | internal val readingWritingState = ReadingWriting(this) |
| 44 | // state transitions |
Sergey Mashkov | 86f7061 | 2017-07-25 10:59:44 +0300 | [diff] [blame] | 45 | override fun startReading() = readingState |
| 46 | override fun startWriting() = writingState |
Sergey Mashkov | 86f7061 | 2017-07-25 10:59:44 +0300 | [diff] [blame] | 47 | override val idle: Boolean get() = error("Not available for initial state") |
Sergey Mashkov | 86f7061 | 2017-07-25 10:59:44 +0300 | [diff] [blame] | 48 | override fun toString() = "Initial" |
| 49 | } |
| 50 | |
Roman Elizarov | 8376203 | 2017-08-08 19:58:40 +0300 | [diff] [blame] | 51 | class IdleNonEmpty internal constructor( |
| 52 | val initial: Initial // public here, so can release initial state when idle |
| 53 | ) : ReadWriteBufferState(initial.backingBuffer, initial.capacity) { |
| 54 | override fun startReading() = initial.readingState |
| 55 | override fun startWriting() = initial.writingState |
Sergey Mashkov | 86f7061 | 2017-07-25 10:59:44 +0300 | [diff] [blame] | 56 | override val idle: Boolean get() = true |
Sergey Mashkov | 86f7061 | 2017-07-25 10:59:44 +0300 | [diff] [blame] | 57 | override fun toString() = "IDLE(with buffer)" |
| 58 | } |
| 59 | |
Roman Elizarov | 8376203 | 2017-08-08 19:58:40 +0300 | [diff] [blame] | 60 | class Reading internal constructor( |
| 61 | private val initial: Initial |
| 62 | ) : ReadWriteBufferState(initial.backingBuffer, initial.capacity) { |
| 63 | override val readBuffer: ByteBuffer get() = initial.readBuffer |
| 64 | override fun startWriting() = initial.readingWritingState |
| 65 | override fun stopReading() = initial.idleState |
Sergey Mashkov | 86f7061 | 2017-07-25 10:59:44 +0300 | [diff] [blame] | 66 | override fun toString() = "Reading" |
| 67 | } |
| 68 | |
Roman Elizarov | 8376203 | 2017-08-08 19:58:40 +0300 | [diff] [blame] | 69 | class Writing internal constructor( |
| 70 | private val initial: Initial |
| 71 | ) : ReadWriteBufferState(initial.backingBuffer, initial.capacity) { |
| 72 | override val writeBuffer: ByteBuffer get() = initial.writeBuffer |
| 73 | override fun startReading() = initial.readingWritingState |
| 74 | override fun stopWriting() = initial.idleState |
Sergey Mashkov | 86f7061 | 2017-07-25 10:59:44 +0300 | [diff] [blame] | 75 | override fun toString() = "Writing" |
| 76 | } |
| 77 | |
Roman Elizarov | 8376203 | 2017-08-08 19:58:40 +0300 | [diff] [blame] | 78 | class ReadingWriting internal constructor( |
| 79 | private val initial: Initial |
| 80 | ) : ReadWriteBufferState(initial.backingBuffer, initial.capacity) { |
| 81 | override val readBuffer: ByteBuffer get() = initial.readBuffer |
| 82 | override val writeBuffer: ByteBuffer get() = initial.writeBuffer |
| 83 | override fun stopReading() = initial.writingState |
| 84 | override fun stopWriting() = initial.readingState |
Sergey Mashkov | 86f7061 | 2017-07-25 10:59:44 +0300 | [diff] [blame] | 85 | override fun toString() = "Reading+Writing" |
| 86 | } |
| 87 | |
Roman Elizarov | 469cad3 | 2017-08-15 15:54:56 +0300 | [diff] [blame] | 88 | object Terminated : ReadWriteBufferState(EmptyByteBuffer, EmptyCapacity) { |
Sergey Mashkov | 86f7061 | 2017-07-25 10:59:44 +0300 | [diff] [blame] | 89 | override fun toString() = "Terminated" |
| 90 | } |
Roman Elizarov | 8376203 | 2017-08-08 19:58:40 +0300 | [diff] [blame] | 91 | } |