blob: 5d6b7bc1f3c909e9a820e89fd047013d740968fe [file] [log] [blame]
// Copyright (c) 2013, the Dart project authors. Please see the AUTHORS file
// for details. All rights reserved. Use of this source code is governed by a
// BSD-style license that can be found in the LICENSE file.
class _IOServicePorts {
List<SendPort> _freeServicePorts = <SendPort>[];
HashMap<int, SendPort> _usedBy = new HashMap<int, SendPort>();
_IOServicePorts();
SendPort _getFreePort(int forRequestId) {
if (_freeServicePorts.isEmpty) {
_freeServicePorts.add(_newServicePort());
}
SendPort freePort = _freeServicePorts.removeLast();
assert(!_usedBy.containsKey(forRequestId));
_usedBy[forRequestId] = freePort;
return freePort;
}
void _returnPort(int forRequestId) {
_freeServicePorts.add(_usedBy.remove(forRequestId));
}
static SendPort _newServicePort() native "IOService_NewServicePort";
}
@patch
class _IOService {
static _IOServicePorts _servicePorts = new _IOServicePorts();
static RawReceivePort _receivePort;
static SendPort _replyToPort;
static HashMap<int, Completer> _messageMap = new HashMap<int, Completer>();
static int _id = 0;
@patch
static Future _dispatch(int request, List data) {
int id;
do {
id = _getNextId();
} while (_messageMap.containsKey(id));
SendPort servicePort = _servicePorts._getFreePort(id);
_ensureInitialize();
var completer = new Completer();
_messageMap[id] = completer;
try {
servicePort.send([id, _replyToPort, request, data]);
} catch (error) {
_messageMap.remove(id).complete(error);
if (_messageMap.length == 0) {
_finalize();
}
}
return completer.future;
}
static void _ensureInitialize() {
if (_receivePort == null) {
_receivePort = new RawReceivePort();
_replyToPort = _receivePort.sendPort;
_receivePort.handler = (data) {
assert(data is List && data.length == 2);
_messageMap.remove(data[0]).complete(data[1]);
_servicePorts._returnPort(data[0]);
if (_messageMap.length == 0) {
_finalize();
}
};
}
}
static void _finalize() {
_id = 0;
_receivePort.close();
_receivePort = null;
}
static int _getNextId() {
if (_id == 0x7FFFFFFF) _id = 0;
return _id++;
}
}