Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(55)

Unified Diff: sdk/lib/isolate/isolate_stream.dart

Issue 11783009: Big merge from experimental to bleeding edge. (Closed) Base URL: https://dart.googlecode.com/svn/branches/bleeding_edge/dart
Patch Set: Created 7 years, 11 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
« no previous file with comments | « sdk/lib/isolate/isolate_sources.gypi ('k') | sdk/lib/isolate/mangler.dart » ('j') | no next file with comments »
Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
Index: sdk/lib/isolate/isolate_stream.dart
diff --git a/sdk/lib/isolate/isolate_stream.dart b/sdk/lib/isolate/isolate_stream.dart
new file mode 100644
index 0000000000000000000000000000000000000000..2641015475408a5e2a2f467b2464b057e3b9d8e9
--- /dev/null
+++ b/sdk/lib/isolate/isolate_stream.dart
@@ -0,0 +1,215 @@
+// Copyright (c) 2012, 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.
+
+part of dart.isolate;
+
+/**
+ * The initial [IsolateStream] available by default for this isolate. This
+ * [IsolateStream] is created automatically and it is commonly used to establish
+ * the first communication between isolates (see [streamSpawnFunction] and
+ * [streamSpawnUri]).
+ */
+final IsolateStream stream = new IsolateStream._fromOriginalReceivePort(port);
+
+/**
+ * A [MessageBox] creates an [IsolateStream], [stream], and an [IsolateSink],
+ * [sink].
+ *
+ * Any message that is written into the [sink] (independent of the isolate) is
+ * sent to the [stream] where its subscribers can react to the messages.
+ */
+class MessageBox {
+ final IsolateStream stream;
+ final IsolateSink sink;
+
+ MessageBox.oneShot() : this._oneShot(new ReceivePort());
+ MessageBox._oneShot(ReceivePort receivePort)
+ : stream = new IsolateStream._fromOriginalReceivePortOneShot(receivePort),
+ sink = new IsolateSink._fromPort(receivePort.toSendPort());
+
+ MessageBox() : this._(new ReceivePort());
+ MessageBox._(ReceivePort receivePort)
+ : stream = new IsolateStream._fromOriginalReceivePort(receivePort),
+ sink = new IsolateSink._fromPort(receivePort.toSendPort());
+}
+
+// Used for mangling.
+const int _ISOLATE_STREAM_TOKEN = 132421119;
+
+class _CloseToken {
+ /// This token is sent from [IsolateSink]s to [IsolateStream]s to ask them to
+ /// close themselves.
+ const _CloseToken();
+}
+
+/**
+ * [IsolateStream]s, together with [IsolateSink]s, are the only means of
+ * communication between isolates. Each IsolateStream has a corresponding
+ * [IsolateSink]. Any message written into that sink will be delivered to
+ * the stream and then dispatched to the stream's subscribers.
+ */
+class IsolateStream extends Stream<dynamic> {
+ bool _isClosed = false;
+ final ReceivePort _port;
+ StreamController _controller = new StreamController();
+
+ IsolateStream._fromOriginalReceivePort(this._port) {
+ _port.receive((message, replyTo) {
+ assert(replyTo == null);
+ _add(message);
+ });
+ }
+
+ IsolateStream._fromOriginalReceivePortOneShot(this._port) {
+ _port.receive((message, replyTo) {
+ assert(replyTo == null);
+ _add(message);
+ close();
+ });
+ }
+
+ void _add(var message) {
+ message = _unmangleMessage(message);
+ if (identical(message, const _CloseToken())) {
+ close();
+ } else {
+ _controller.sink.add(message);
+ }
+ }
+
+ /**
+ * Close the stream from the receiving end.
+ *
+ * Closing an already closed port has no effect.
+ */
+ void close() {
+ if (!_isClosed) {
+ _isClosed = true;
+ _port.close();
+ _controller.close();
+ }
+ }
+
+ StreamSubscription<T> listen(void onData(T event),
+ { void onError(AsyncError error),
+ void onDone(),
+ bool unsubscribeOnError}) {
+ return _controller.listen(onData,
+ onError: onError,
+ onDone: onDone,
+ unsubscribeOnError: unsubscribeOnError);
+ }
+
+ dynamic _unmangleMessage(var message) {
+ _IsolateDecoder decoder = new _IsolateDecoder(
+ _ISOLATE_STREAM_TOKEN,
+ (data) {
+ if (data is! List) return data;
+ if (data.length == 2 && data[0] == "Sink" && data[1] is SendPort) {
+ return new IsolateSink._fromPort(data[1]);
+ }
+ if (data.length == 1 && data[0] == "Close") {
+ return const _CloseToken();
+ }
+ return data;
+ });
+ return decoder.decode(message);
+ }
+}
+
+/**
+ * [IsolateSink]s represent the feed for [IsolateStream]s. Any message written
+ * to [this] is delivered to its respective [IsolateStream]. [IsolateSink]s are
+ * created by [MessageBox]es.
+ *
+ * [IsolateSink]s can be transmitted to other isolates.
+ */
+class IsolateSink extends StreamSink<dynamic> {
+ bool _isClosed = false;
+ final SendPort _port;
+ IsolateSink._fromPort(this._port);
+
+ /**
+ * Sends an asynchronous [message] to the linked [IsolateStream]. The message
+ * is copied to the receiving isolate.
+ *
+ * The content of [message] can be: primitive values (null, num, bool, double,
+ * String), instances of [IsolateSink]s, and lists and maps whose elements are
+ * any of these. List and maps are also allowed to be cyclic.
+ *
+ * In the special circumstances when two isolates share the same code and are
+ * running in the same process (e.g. isolates created via [spawnFunction]), it
+ * is also possible to send object instances (which would be copied in the
+ * process). This is currently only supported by the dartvm. For now, the
+ * dart2js compiler only supports the restricted messages described above.
+ */
+ void add(dynamic message) {
+ var mangled = _mangleMessage(message);
+ _port.send(mangled);
+ }
+
+ void signalError(AsyncError errorEvent) {
+ throw new UnimplementedError("signalError on isolate streams");
+ }
+
+ dynamic _mangleMessage(var message) {
+ _IsolateEncoder encoder = new _IsolateEncoder(
+ _ISOLATE_STREAM_TOKEN,
+ (data) {
+ if (data is IsolateSink) return ["Sink", data._port];
+ if (identical(data, const _CloseToken())) return ["Close"];
+ return data;
+ });
+ return encoder.encode(message);
+ }
+
+ void close() {
+ if (_isClosed) throw new StateError("Sending on closed stream");
+ add(const _CloseToken());
+ _isClosed = true;
+ }
+
+ /**
+ * Tests whether [other] is an [IsolateSink] feeding into the same
+ * [IsolateStream] as this one.
+ */
+ bool operator==(var other) {
+ return other is IsolateSink && _port == other._port;
+ }
+
+ int get hashCode => _port.hashCode + 499;
+}
+
+
+/**
+ * Creates and spawns an isolate that shares the same code as the current
+ * isolate, but that starts from [topLevelFunction]. The [topLevelFunction]
+ * argument must be a static top-level function or a static method that takes no
+ * arguments.
+ *
+ * When any isolate starts (even the main script of the application), a default
+ * [IsolateStream] is created for it. This sink is available from the top-level
+ * getter [stream] defined in this library.
+ *
+ * [spawnFunction] returns an [IsolateSink] feeding into the child isolate's
+ * default stream.
+ *
+ * See comments at the top of this library for more details.
+ */
+IsolateSink streamSpawnFunction(void topLevelFunction()) {
+ SendPort sendPort = spawnFunction(topLevelFunction);
+ return new IsolateSink._fromPort(sendPort);
+}
+
+/**
+ * Creates and spawns an isolate whose code is available at [uri]. Like with
+ * [streamSpawnFunction], the child isolate will have a default [IsolateStream],
+ * and a this function returns an [IsolateSink] feeding into it.
+ *
+ * See comments at the top of this library for more details.
+ */
+IsolateSink streamSpawnUri(String uri) {
+ SendPort sendPort = spawnUri(uri);
+ return new IsolateSink._fromPort(sendPort);
+}
« no previous file with comments | « sdk/lib/isolate/isolate_sources.gypi ('k') | sdk/lib/isolate/mangler.dart » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698