-
Notifications
You must be signed in to change notification settings - Fork 70
/
stream_info_impl.h
263 lines (217 loc) · 8.63 KB
/
stream_info_impl.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
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
#ifndef STREAM_INFO_IMPL_H
#define STREAM_INFO_IMPL_H
#include "common.h"
#include <cstdint>
#include <mutex>
#include <pugixml.hpp>
#include <string>
#include <unordered_map>
namespace lsl {
/// LRU cache for queries
class query_cache {
std::unordered_map<std::string, int> cache;
int query_cache_age{0};
std::mutex cache_mut_;
public:
bool matches_query(const pugi::xml_document &doc, const std::string &query, bool nocache);
};
/**
* Actual implementation of the stream_info class.
*
* The stream_info class forwards all operations to an instance of this class.
*/
class stream_info_impl {
public:
stream_info_impl();
/**
* @brief Construct a new stream_info object.
* Here is where the core format and identity information of a stream is specified.
* @param name Name of the stream (usually a CamelCase word).
* Describes the device (or product series) that this stream makes available (for use by
* programs, experimenters, or data analysts).
* @param type Content type of the stream. Please see https://github.com/sccn/xdf/wiki/Meta-Data
* (or web search for: XDF meta-data) for pre-defined content-type names, but you can also make
* up your own.
* @param channel_count Number of channels in a sample. Each channel in a sample has the same
* format and each sample in a stream has the same number of channels.
* @param nominal_srate The sampling rate (in Hz) advertised by the device, if regular
* (otherwise set to ::IRREGULAR_RATE).
* @param channel_format Format/type of each channel. If your channels have different formats,
* use the largest type that can hold them all (e.g., double) or consider supplying multiple
* streams.
* @param source_id Unique identifier of the source or device if available (e.g., serial
* number). Allows for advanced failure recovery.
*/
stream_info_impl(const std::string &name, std::string type, int channel_count,
double nominal_srate, lsl_channel_format_t channel_format, std::string source_id);
/// Copy constructor. Needs special handling because xml_document is non-copyable.
stream_info_impl(const stream_info_impl &rhs);
/// Assignment operator. Needs extra handling because xml_document is non-copyable.
stream_info_impl &operator=(const stream_info_impl &rhs);
// === Protocol Support Operations ===
/**
* Get the short-info message according to this stream_info.
*
* The short-info message is a shortened xml representation of the stream_info, excluding the
* .desc() field (which can be megabytes in size).
* This message is sent by a stream outlet in response to a variety of queries.
*/
std::string to_shortinfo_message();
/**
* Initialize a stream_info from a short-info message.
*
* This functions resets all fields of the stream_info accoridng to the message. The .desc()
* field will be empty.
*/
void from_shortinfo_message(const std::string &m);
/**
* Get the full-info message for this stream_info.
*
* This is a complete XML representation of the stream_info.
*/
std::string to_fullinfo_message();
/**
* Initialize a stream_info from a full-info message.
*
* This functions resets all fields of the stream_info accoridng to the message.
*/
void from_fullinfo_message(const std::string &m);
/**
* Test whether this stream info matches the given query string.
*
* The info "matches" if the given XPath 1.0 query string returns a non-empty node set.
* @return Whether stream info is matched by the query string.
*/
bool matches_query(const std::string &query, bool nocache = false);
//
// === Data Information Getters ===
//
/// Get the name of a stream.
const std::string &name() const { return name_; }
/// Get the content type of the stream.
const std::string &type() const { return type_; }
/// Get the number of channels of a stream.
uint32_t channel_count() const { return channel_count_; }
/// Get the sampling rate of a stream (in Hz) as advertised by the device.
double nominal_srate() const { return nominal_srate_; }
/// Get the channel format of a stream.
lsl_channel_format_t channel_format() const { return channel_format_; }
/// Get the unique source identifier of a stream, if any.
const std::string &source_id() const { return source_id_; }
/// Get the number of bytes per channel (returns 0 for string-typed channels).
int channel_bytes() const;
/// Get the number of bytes per sample (returns 0 for string-typed channels).
int sample_bytes() const { return channel_count_ * channel_bytes(); }
//
// === Network Identity Information Getters/Setters ===
//
/// Get/Set the info / protocol version used by the stream.
int version() const { return version_; }
void version(int v);
/**
* Get/Set the creation time stamp of a stream.
*
* This is the time stamp (via local_clock()) of when the stream was first created (in the time
* domain of the providing machine).
*/
double created_at() const { return created_at_; }
void created_at(double v);
/**
* Get/Set the UID of a stream instance (once assigned).
*
* This is a unique identifier of the stream instance, and is guaranteed to be different across
* multiple instantiations of the same stream (e.g., after a re-start).
*/
const std::string &uid() const { return uid_; }
void uid(const std::string &v);
/// Reset the UID to a randomly generated UUID4
const std::string &reset_uid();
/**
* Get/Set the session id for the given stream.
*
* The session ID is an optional human-assigned identifier of the recording session; only inlets
* and outlets that have the same session id can be paired with each other to avoid accidentally
* recording from an unrelated concurrent session on the same network.
* The session id can be set via the configuration file (api_config::session_id).
*/
const std::string &session_id() const { return session_id_; }
void session_id(const std::string &v);
/// Get/Set the provider hostname for the given stream.
const std::string &hostname() const { return hostname_; }
void hostname(const std::string &v);
/**
* Get/Set the host name or IP address where the stream is hosted.
*
* This may be a fully resolved address (such as testing.uscd.edu) or an IPv4 or IPv6
* address in string form.
*/
const std::string &v4address() const { return v4address_; }
void v4address(const std::string &v);
/**
* @brief Get/Set the TCP data port where the stream is hosted (once assigned).
* This port is internally used to obtain data and meta-data from a stream.
*/
uint16_t v4data_port() const { return v4data_port_; }
void v4data_port(uint16_t v);
/**
* Get/Set the UDP service port where the stream is hosted (once assigned).
* This port is internally used to obtain time correction information for a stream.
*/
uint16_t v4service_port() const { return v4service_port_; }
void v4service_port(uint16_t v);
/**
* Get/Set the host name or IP address where the stream is hosted.
* This may be a fully resolved address (such as testing.uscd.edu) or an IPv4 or IPv6 address in
* string form.
*/
const std::string &v6address() const { return v6address_; }
void v6address(const std::string &v);
/**
* Get/Set the TCP data port where the stream is hosted (once assigned).
* This port is internally used to obtain data and meta-data from a stream.
*/
uint16_t v6data_port() const { return v6data_port_; }
void v6data_port(uint16_t v);
/**
* Get/Set the UDP service port where the stream is hosted (once assigned).
* This port is internally used to obtain time correction information for a stream.
*/
uint16_t v6service_port() const { return v6service_port_; }
void v6service_port(uint16_t v);
/// Get the (editable) XML description of a stream.
pugi::xml_node desc();
pugi::xml_node desc() const;
/// helper function to calculate the buffer size in samples for inlets and outlets
uint32_t calc_transport_buf_samples(int32_t requested_len, lsl_transport_options_t flags) const;
protected:
/// Create and assign the XML DOM structure based on the class fields.
void write_xml(pugi::xml_document &doc);
/// Read the class fields from an XML DOM structure.
void read_xml(pugi::xml_document &doc);
private:
// data information
std::string name_;
std::string type_;
uint32_t channel_count_;
double nominal_srate_;
lsl_channel_format_t channel_format_;
std::string source_id_;
// auto-generated network information
int version_;
std::string v4address_;
uint16_t v4data_port_;
uint16_t v4service_port_;
std::string v6address_;
uint16_t v6data_port_;
uint16_t v6service_port_;
std::string uid_;
double created_at_;
std::string session_id_;
std::string hostname_;
// XML representation
pugi::xml_document doc_;
// cached query results
query_cache cached_;
};
} // namespace lsl
#endif