Chromium Code Reviews| OLD | NEW |
|---|---|
| 1 // Copyright 2014 The Chromium Authors. All rights reserved. | 1 // Copyright 2014 The Chromium Authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 part of bindings; | 5 part of bindings; |
| 6 | 6 |
| 7 abstract class Client { | 7 abstract class Client { |
| 8 core.MojoMessagePipeEndpoint _endpoint; | 8 core.MojoMessagePipeEndpoint _endpoint; |
| 9 core.MojoHandle _handle; | 9 core.MojoHandle _handle; |
| 10 List _sendQueue; | 10 List _sendQueue; |
| 11 List _completerQueue; | 11 Map<int, Completer> _completerMap; |
| 12 bool _isOpen = false; | 12 bool _isOpen = false; |
| 13 int _nextId = 0; | |
| 13 | 14 |
| 14 void handleResponse(MessageReader reader); | 15 void handleResponse(MessageReader reader); |
| 15 | 16 |
| 16 Client(this._endpoint) { | 17 Client(core.MojoMessagePipeEndpoint endpoint) : |
| 18 _sendQueue = [], | |
| 19 _completerMap = {}, | |
| 20 _endpoint = endpoint, | |
| 21 _handle = new core.MojoHandle(endpoint.handle); | |
| 22 | |
| 23 Client.fromHandle(int handle) { | |
| 17 _sendQueue = []; | 24 _sendQueue = []; |
| 18 _completerQueue = []; | 25 _completerMap = {}; |
| 26 _endpoint = | |
| 27 new core.MojoMessagePipeEndpoint(new core.RawMojoHandle(handle)); | |
| 19 _handle = new core.MojoHandle(_endpoint.handle); | 28 _handle = new core.MojoHandle(_endpoint.handle); |
| 20 } | 29 } |
| 21 | 30 |
| 31 void _doRead() { | |
| 32 // Query how many bytes are available. | |
| 33 var result = _endpoint.query(); | |
| 34 if (!result.status.isOk && !result.status.isResourceExhausted) { | |
| 35 // If something else happens, it means the handle wasn't really ready | |
| 36 // for reading, which indicates a bug in MojoHandle or the | |
| 37 // handle watcher. | |
| 38 throw "message pipe query failed: ${result.status}"; | |
| 39 } | |
| 40 | |
| 41 // Read the data. | |
| 42 var bytes = new ByteData(result.bytesRead); | |
| 43 var handles = new List<core.RawMojoHandle>(result.handlesRead); | |
| 44 result = _endpoint.read(bytes, result.bytesRead, handles); | |
| 45 if (!result.status.isOk && !result.status.isResourceExhausted) { | |
| 46 throw "message pipe read failed: ${result.status}"; | |
| 47 } | |
| 48 var message = new Message(bytes, handles); | |
| 49 var reader = new MessageReader(message); | |
| 50 | |
| 51 handleResponse(reader); | |
| 52 } | |
| 53 | |
| 54 void _doWrite() { | |
| 55 if (_sendQueue.length > 0) { | |
| 56 List messageCompleter = _sendQueue.removeAt(0); | |
| 57 Message message = messageCompleter[0]; | |
| 58 Completer completer = messageCompleter[1]; | |
| 59 _endpoint.write(message.buffer, | |
| 60 message.buffer.lengthInBytes, | |
| 61 message.handles); | |
| 62 if (!_endpoint.status.isOk) { | |
| 63 throw "message pipe write failed"; | |
| 64 } | |
| 65 if (completer != null) { | |
| 66 if (!message.expectsResponse) { | |
| 67 throw "Message has a completer, but does not expect a response"; | |
| 68 } | |
| 69 int requestID = message.requestID; | |
| 70 if (_completerMap[requestID] != null) { | |
| 71 throw "Request ID $requestID is already in use."; | |
| 72 } | |
| 73 _completerMap[requestID] = completer; | |
| 74 } | |
| 75 } | |
| 76 } | |
| 77 | |
| 22 void open() { | 78 void open() { |
| 23 _handle.listen((int mojoSignal) { | 79 _handle.listen((List<int> event) { |
| 24 if (core.MojoHandleSignals.isReadable(mojoSignal)) { | 80 var signalsWatched = new core.MojoHandleSignals(event[0]); |
| 25 // Query how many bytes are available. | 81 var signalsReceived = new core.MojoHandleSignals(event[1]); |
| 26 var result = _endpoint.query(); | 82 if (signalsReceived.isPeerClosed) { |
| 27 if (!result.status.isOk && !result.status.isResourceExhausted) { | 83 close(); |
| 28 // If something else happens, it means the handle wasn't really ready | 84 return; |
| 29 // for reading, which indicates a bug in MojoHandle or the | 85 } |
| 30 // handle watcher. | |
| 31 throw new Exception("message pipe query failed: ${result.status}"); | |
| 32 } | |
| 33 | 86 |
| 34 // Read the data. | 87 if (signalsReceived.isReadable) { |
| 35 var bytes = new ByteData(result.bytesRead); | 88 _doRead(); |
| 36 var handles = new List<core.RawMojoHandle>(result.handlesRead); | 89 } |
| 37 result = _endpoint.read(bytes, result.bytesRead, handles); | |
| 38 if (!result.status.isOk && !result.status.isResourceExhausted) { | |
| 39 throw new Exception("message pipe read failed: ${result.status}"); | |
| 40 } | |
| 41 var message = new Message(bytes, handles); | |
| 42 var reader = new MessageReader(message); | |
| 43 | 90 |
| 44 handleResponse(reader); | 91 if (signalsReceived.isWritable) { |
| 92 _doWrite(); | |
| 45 } | 93 } |
| 46 if (core.MojoHandleSignals.isWritable(mojoSignal)) { | 94 |
| 47 if (_sendQueue.length > 0) { | 95 if (_sendQueue.length == 0) { |
| 48 List messageCompleter = _sendQueue.removeAt(0); | 96 var withoutWritable = signalsWatched - core.MojoHandleSignals.WRITABLE; |
| 49 _endpoint.write(messageCompleter[0].buffer, | 97 _handle.enableSignals(withoutWritable); |
| 50 messageCompleter[0].buffer.lengthInBytes, | 98 } else { |
| 51 messageCompleter[0].handles); | 99 _handle.enableSignals(signalsWatched); |
| 52 if (!_endpoint.status.isOk) { | |
| 53 throw new Exception("message pipe write failed"); | |
| 54 } | |
| 55 if (messageCompleter[1] != null) { | |
| 56 _completerQueue.add(messageCompleter[1]); | |
| 57 } | |
| 58 } | |
| 59 if ((_sendQueue.length == 0) && _handle.writeEnabled()) { | |
| 60 _handle.disableWriteEvents(); | |
| 61 } | |
| 62 } | |
| 63 if (core.MojoHandleSignals.isNone(mojoSignal)) { | |
| 64 // The handle watcher will send MojoHandleSignals.NONE if the other | |
| 65 // endpoint of the pipe is closed. | |
| 66 _handle.close(); | |
| 67 } | 100 } |
| 68 }); | 101 }); |
| 69 _isOpen = true; | 102 _isOpen = true; |
| 70 } | 103 } |
| 71 | 104 |
| 72 void close() { | 105 void close() { |
| 73 assert(isOpen); | 106 if (_isOpen) { |
| 74 _handle.close(); | 107 _handle.close(); |
| 75 _isOpen = false; | 108 _handle = null; |
| 109 _isOpen = false; | |
| 110 } | |
| 76 } | 111 } |
| 77 | 112 |
| 78 void enqueueMessage(Type t, int name, Object msg) { | 113 void enqueueMessage(Type t, int name, Object msg) { |
| 79 var builder = new MessageBuilder(name, align(getEncodedSize(t))); | 114 var builder = new MessageBuilder(name, align(getEncodedSize(t))); |
| 80 builder.encodeStruct(t, msg); | 115 builder.encodeStruct(t, msg); |
| 81 var message = builder.finish(); | 116 var message = builder.finish(); |
| 82 _sendQueue.add([message, null]); | 117 _sendQueue.add([message, null]); |
| 83 if (!_handle.writeEnabled()) { | 118 if (_sendQueue.length == 1) { |
| 84 _handle.enableWriteEvents(); | 119 _handle.enableAllEvents(); |
| 85 } | 120 } |
| 86 } | 121 } |
| 87 | 122 |
| 123 int _getNextId() { | |
| 124 int next = _nextId; | |
| 125 _nextId++; | |
| 126 if (_nextId.bitLength > 64) { | |
|
siva
2014/12/29 23:20:44
maybe document what this magic number 64 is.
zra
2014/12/30 16:29:33
Added a comment.
| |
| 127 _nextId = 1; | |
| 128 } | |
| 129 return next; | |
| 130 } | |
| 131 | |
| 88 Future enqueueMessageWithRequestID( | 132 Future enqueueMessageWithRequestID( |
| 89 Type t, int name, int id, int flags, Object msg) { | 133 Type t, int name, int id, int flags, Object msg) { |
| 134 if (id == -1) { | |
| 135 id = _getNextId(); | |
| 136 } | |
| 137 | |
| 90 var builder = new MessageWithRequestIDBuilder( | 138 var builder = new MessageWithRequestIDBuilder( |
| 91 name, align(getEncodedSize(t)), id, flags); | 139 name, align(getEncodedSize(t)), id, flags); |
| 92 builder.encodeStruct(t, msg); | 140 builder.encodeStruct(t, msg); |
| 93 var message = builder.finish(); | 141 var message = builder.finish(); |
| 94 | 142 |
| 95 var completer = new Completer(); | 143 var completer = new Completer(); |
| 96 _sendQueue.add([message, completer]); | 144 _sendQueue.add([message, completer]); |
| 97 if (!_handle.writeEnabled()) { | 145 if (_sendQueue.length == 1) { |
| 98 _handle.enableWriteEvents(); | 146 _handle.enableAllEvents(); |
| 147 } else { | |
| 148 _handle.enableReadEvents(); | |
| 99 } | 149 } |
| 100 return completer.future; | 150 return completer.future; |
| 101 } | 151 } |
| 102 | 152 |
| 103 // Need a getter for this for access in subclasses. | 153 // Need a getter for this for access in subclasses. |
| 104 List get completerQueue => _completerQueue; | 154 Map<int, Completer> get completerMap => _completerMap; |
| 105 bool get isOpen => _isOpen; | 155 bool get isOpen => _isOpen; |
| 106 } | 156 } |
| OLD | NEW |