-
Notifications
You must be signed in to change notification settings - Fork 1.6k
/
Copy pathport.h
149 lines (113 loc) · 4.34 KB
/
port.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
// Copyright (c) 2011, 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.
#ifndef RUNTIME_VM_PORT_H_
#define RUNTIME_VM_PORT_H_
#include <memory>
#include "include/dart_api.h"
#include "vm/allocation.h"
#include "vm/globals.h"
#include "vm/json_stream.h"
#include "vm/lockers.h"
#include "vm/port_set.h"
#include "vm/random.h"
namespace dart {
class Isolate;
class Message;
class MessageHandler;
class Mutex;
class PortHandler;
class PortMap : public AllStatic {
public:
// Allocate a port for the provided handler and return its VM-global id.
static Dart_Port CreatePort(PortHandler* handler);
// Close the port with id. All pending messages will be dropped.
//
// Returns true if the port is successfully closed.
static bool ClosePort(Dart_Port id, PortHandler** port_handler = nullptr);
// Close all the ports for the provided handler.
static void ClosePorts(MessageHandler* handler);
// Enqueues the message in the port with id. Returns false if the port is not
// active any longer.
//
// Claims ownership of 'message'.
static bool PostMessage(std::unique_ptr<Message> message,
bool before_events = false);
// Returns the origin id for port 'id'.
static Dart_Port GetOriginId(Dart_Port id);
// Returns whether the isolate that owns the port is owned by the current
// thread.
static bool IsOwnedByCurrentThread(Dart_Port id);
#if defined(TESTING)
static Isolate* GetIsolate(Dart_Port id);
static bool PortExists(Dart_Port id);
static bool HasPorts(MessageHandler* handler);
#endif
// Whether the destination port's isolate is a member of [isolate_group].
static bool IsReceiverInThisIsolateGroupOrClosed(Dart_Port receiver,
IsolateGroup* group);
static void Init();
static void Shutdown();
static void Cleanup();
static void PrintPortsForMessageHandler(MessageHandler* handler,
JSONStream* stream);
static void DebugDumpForMessageHandler(MessageHandler* handler);
class Locker : public MutexLocker {
public:
Locker() : MutexLocker(PortMap::mutex_) {}
};
private:
struct Entry : public PortSet<Entry>::Entry {
Entry() : handler(nullptr) {}
Entry(Dart_Port port, PortHandler* handler)
: PortSet<Entry>::Entry(port), handler(handler) {}
PortHandler* handler;
};
// Allocate a new unique port.
static Dart_Port AllocatePort();
static Isolate* GetIsolateLocked(const Locker& ml, Dart_Port id);
// Lock protecting access to the port map.
static Mutex* mutex_;
static PortSet<Entry>* ports_;
static Random* prng_;
};
// An object handling messages dispatched to one or more ports in the |PortMap|.
class PortHandler {
public:
virtual ~PortHandler();
virtual const char* name() const = 0;
// Notify the handler that a port previously associated with it is
// now closed.
virtual void OnPortClosed(Dart_Port port) = 0;
#if defined(DEBUG)
// Check that it is safe to access this port handler.
//
// For example, if this |PortHandler| is an isolate, then it is
// only safe to access it when it is the current isolate.
virtual void CheckAccess() const;
#endif
// Return Isolate to which this message handler corresponds to.
virtual Isolate* isolate() const = 0;
// Ask the handler to shutdown, e.g. stop associated thread pools if any.
virtual void Shutdown() = 0;
// Posts a message on this handler's message queue.
// If before_events is true, then the message is enqueued before any pending
// events, but after any pending isolate library events.
virtual void PostMessage(std::unique_ptr<Message> message,
bool before_events = false) = 0;
protected:
struct PortSetEntry : public PortSet<PortSetEntry>::Entry {
PortSetEntry() : Entry() {}
explicit PortSetEntry(Dart_Port port) : Entry(port) {}
};
private:
friend class PortMap;
// Returns set of ports associate with this handler if
// handler supports multiple ports or |nullptr| otherwise.
//
// Only |PortMap| is expected to call this method under locked
// PortMap::mutex_.
virtual PortSet<PortSetEntry>* ports(PortMap::Locker& locker) = 0;
};
} // namespace dart
#endif // RUNTIME_VM_PORT_H_