blob: 274fa478ac2b6855b02575ac313a663a51ba379c [file] [log] [blame]
// Copyright 2014 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
part of bindings;
abstract class Client {
core.MojoMessagePipeEndpoint _endpoint;
core.MojoHandle _handle;
List _sendQueue;
List _completerQueue;
bool _isOpen = false;
void handleResponse(MessageReader reader);
Client(this._endpoint) {
_sendQueue = [];
_completerQueue = [];
_handle = new core.MojoHandle(_endpoint.handle);
}
void open() {
_handle.listen((int mojoSignal) {
if (core.MojoHandleSignals.isReadable(mojoSignal)) {
// Query how many bytes are available.
var result = _endpoint.query();
if (!result.status.isOk && !result.status.isResourceExhausted) {
// If something else happens, it means the handle wasn't really ready
// for reading, which indicates a bug in MojoHandle or the
// handle watcher.
throw new Exception("message pipe query failed: ${result.status}");
}
// Read the data.
var bytes = new ByteData(result.bytesRead);
var handles = new List<core.RawMojoHandle>(result.handlesRead);
result = _endpoint.read(bytes, result.bytesRead, handles);
if (!result.status.isOk && !result.status.isResourceExhausted) {
throw new Exception("message pipe read failed: ${result.status}");
}
var message = new Message(bytes, handles);
var reader = new MessageReader(message);
handleResponse(reader);
}
if (core.MojoHandleSignals.isWritable(mojoSignal)) {
if (_sendQueue.length > 0) {
List messageCompleter = _sendQueue.removeAt(0);
_endpoint.write(messageCompleter[0].buffer);
if (!_endpoint.status.isOk) {
throw new Exception("message pipe write failed");
}
if (messageCompleter[1] != null) {
_completerQueue.add(messageCompleter[1]);
}
}
if ((_sendQueue.length == 0) && _handle.writeEnabled()) {
_handle.disableWriteEvents();
}
}
if (core.MojoHandleSignals.isNone(mojoSignal)) {
// The handle watcher will send MojoHandleSignals.NONE if the other
// endpoint of the pipe is closed.
_handle.close();
}
});
_isOpen = true;
}
void close() {
assert(isOpen);
_handle.close();
_isOpen = false;
}
void enqueueMessage(Type t, int name, Object msg) {
var builder = new MessageBuilder(name, align(getEncodedSize(t)));
builder.encodeStruct(t, msg);
var message = builder.finish();
_sendQueue.add([message, null]);
if ((_sendQueue.length > 0) && !_handle.writeEnabled()) {
_handle.enableWriteEvents();
}
}
Future enqueueMessageWithRequestID(Type t, int name, int id, Object msg) {
var builder = new MessageWithRequestIDBuilder(
name, align(getEncodedSize(t)), id);
builder.encodeStruct(t, msg);
var message = builder.finish();
var completer = new Completer();
_sendQueue.add([message, completer]);
if ((_sendQueue.length > 0) && !_handle.writeEnabled()) {
_handle.enableWriteEvents();
}
return completer.future;
}
// Need a getter for this for access in subclasses.
List get completerQueue => _completerQueue;
bool get isOpen => _isOpen;
}