1 # The MIT License (MIT)
3 # Copyright (c) 2017 Philippe Proulx <pproulx@efficios.com>
4 # Copyright (c) 2018 Francis Deslauriers <francis.deslauriers@efficios.com>
5 # Copyright (c) 2019 Simon Marchi <simon.marchi@efficios.com>
7 # Permission is hereby granted, free of charge, to any person obtaining a copy
8 # of this software and associated documentation files (the "Software"), to deal
9 # in the Software without restriction, including without limitation the rights
10 # to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
11 # copies of the Software, and to permit persons to whom the Software is
12 # furnished to do so, subject to the following conditions:
14 # The above copyright notice and this permission notice shall be included in
15 # all copies or substantial portions of the Software.
17 # THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
18 # IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
19 # FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
20 # AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
21 # LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
22 # OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
25 __all__
= ['_TraceClass']
28 from bt2
import native_bt
, utils
, object
29 import bt2
.stream_class
31 import collections
.abc
35 class _TraceClassEnv(collections
.abc
.MutableMapping
):
36 def __init__(self
, trace_class
):
37 self
._trace
_class
= trace_class
39 def __getitem__(self
, key
):
42 borrow_entry_fn
= native_bt
.trace_class_borrow_environment_entry_value_by_name_const
43 value_ptr
= borrow_entry_fn(self
._trace
_class
._ptr
, key
)
48 return bt2
.value
._create
_from
_ptr
_and
_get
_ref
(value_ptr
)
50 def __setitem__(self
, key
, value
):
51 if isinstance(value
, str):
52 set_env_entry_fn
= native_bt
.trace_class_set_environment_entry_string
53 elif isinstance(value
, int):
54 set_env_entry_fn
= native_bt
.trace_class_set_environment_entry_integer
56 raise TypeError('expected str or int, got {}'.format(type(value
)))
58 ret
= set_env_entry_fn(self
._trace
_class
._ptr
, key
, value
)
60 utils
._handle
_ret
(ret
, "cannot set trace class object's environment entry")
62 def __delitem__(self
, key
):
63 raise NotImplementedError
66 count
= native_bt
.trace_class_get_environment_entry_count(self
._trace
_class
._ptr
)
71 trace_class_ptr
= self
._trace
_class
_env
._trace
_class
._ptr
73 for idx
in range(len(self
)):
74 borrow_entry_fn
= native_bt
.trace_class_borrow_environment_entry_by_index_const
75 entry_name
, _
= borrow_entry_fn(trace_class_ptr
, idx
)
76 assert entry_name
is not None
80 class _StreamClassIterator(collections
.abc
.Iterator
):
81 def __init__(self
, trace_class
):
82 self
._trace
_class
= trace_class
86 if self
._at
== len(self
._trace
_class
):
89 borrow_stream_class_fn
= native_bt
.trace_class_borrow_stream_class_by_index_const
90 sc_ptr
= borrow_stream_class_fn(self
._trace
_class
._ptr
, self
._at
)
92 id = native_bt
.stream_class_get_id(sc_ptr
)
98 def _trace_class_destruction_listener_from_native(user_listener
, trace_class_ptr
):
99 trace_class
= bt2
.trace_class
._TraceClass
._create
_from
_ptr
_and
_get
_ref
(trace_class_ptr
)
100 user_listener(trace_class
)
103 class _TraceClass(object._SharedObject
, collections
.abc
.Mapping
):
104 _get_ref
= staticmethod(native_bt
.trace_class_get_ref
)
105 _put_ref
= staticmethod(native_bt
.trace_class_put_ref
)
109 uuid_bytes
= native_bt
.trace_class_get_uuid(self
._ptr
)
110 if uuid_bytes
is None:
113 return uuidp
.UUID(bytes
=uuid_bytes
)
115 def _uuid(self
, uuid
):
116 utils
._check
_type
(uuid
, uuidp
.UUID
)
117 native_bt
.trace_class_set_uuid(self
._ptr
, uuid
.bytes
)
119 _uuid
= property(fset
=_uuid
)
121 # Instantiate a trace of this class.
123 def __call__(self
, name
=None):
124 trace_ptr
= native_bt
.trace_create(self
._ptr
)
126 if trace_ptr
is None:
127 raise bt2
.CreationError('cannot create trace class object')
129 trace
= bt2
.trace
._Trace
._create
_from
_ptr
(trace_ptr
)
136 # Number of stream classes in this trace class.
139 count
= native_bt
.trace_class_get_stream_class_count(self
._ptr
)
143 # Get a stream class by stream id.
145 def __getitem__(self
, key
):
146 utils
._check
_uint
64(key
)
148 sc_ptr
= native_bt
.trace_class_borrow_stream_class_by_id_const(self
._ptr
, key
)
152 return bt2
.stream_class
._StreamClass
._create
_from
_ptr
_and
_get
_ref
(sc_ptr
)
155 for idx
in range(len(self
)):
156 sc_ptr
= native_bt
.trace_class_borrow_stream_class_by_index_const(self
._ptr
, idx
)
157 assert sc_ptr
is not None
159 id = native_bt
.stream_class_get_id(sc_ptr
)
166 return _TraceClassEnv(self
)
168 def create_stream_class(self
, id=None,
170 packet_context_field_class
=None,
171 event_common_context_field_class
=None,
172 default_clock_class
=None,
173 assigns_automatic_event_class_id
=True,
174 assigns_automatic_stream_id
=True,
175 packets_have_beginning_default_clock_snapshot
=False,
176 packets_have_end_default_clock_snapshot
=False,
177 supports_discarded_events
=False,
178 discarded_events_have_default_clock_snapshots
=False,
179 supports_discarded_packets
=False,
180 discarded_packets_have_default_clock_snapshots
=False):
182 if self
.assigns_automatic_stream_class_id
:
184 raise ValueError('id provided, but trace class assigns automatic stream class ids')
186 sc_ptr
= native_bt
.stream_class_create(self
._ptr
)
189 raise ValueError('id not provided, but trace class does not assign automatic stream class ids')
191 utils
._check
_uint
64(id)
192 sc_ptr
= native_bt
.stream_class_create_with_id(self
._ptr
, id)
194 sc
= bt2
.stream_class
._StreamClass
._create
_from
_ptr
(sc_ptr
)
199 if packet_context_field_class
is not None:
200 sc
._packet
_context
_field
_class
= packet_context_field_class
202 if event_common_context_field_class
is not None:
203 sc
._event
_common
_context
_field
_class
= event_common_context_field_class
205 if default_clock_class
is not None:
206 sc
._default
_clock
_class
= default_clock_class
208 sc
._assigns
_automatic
_event
_class
_id
= assigns_automatic_event_class_id
209 sc
._assigns
_automatic
_stream
_id
= assigns_automatic_stream_id
210 sc
._packets
_have
_beginning
_default
_clock
_snapshot
= packets_have_beginning_default_clock_snapshot
211 sc
._packets
_have
_end
_default
_clock
_snapshot
= packets_have_end_default_clock_snapshot
212 sc
._set
_supports
_discarded
_events
(supports_discarded_events
,
213 discarded_events_have_default_clock_snapshots
)
214 sc
._set
_supports
_discarded
_packets
(supports_discarded_packets
,
215 discarded_packets_have_default_clock_snapshots
)
219 def assigns_automatic_stream_class_id(self
):
220 return native_bt
.trace_class_assigns_automatic_stream_class_id(self
._ptr
)
222 def _assigns_automatic_stream_class_id(self
, auto_id
):
223 utils
._check
_bool
(auto_id
)
224 return native_bt
.trace_class_set_assigns_automatic_stream_class_id(self
._ptr
, auto_id
)
226 _assigns_automatic_stream_class_id
= property(fset
=_assigns_automatic_stream_class_id
)
228 # Field class creation methods.
230 def _check_create_status(self
, ptr
, type_name
):
232 raise bt2
.CreationError(
233 'cannot create {} field class'.format(type_name
))
235 def _create_integer_field_class(self
, create_func
, py_cls
, type_name
, field_value_range
, preferred_display_base
):
236 field_class_ptr
= create_func(self
._ptr
)
237 self
._check
_create
_status
(field_class_ptr
, type_name
)
239 field_class
= py_cls
._create
_from
_ptr
(field_class_ptr
)
241 if field_value_range
is not None:
242 field_class
._field
_value
_range
= field_value_range
244 if preferred_display_base
is not None:
245 field_class
._preferred
_display
_base
= preferred_display_base
249 def create_signed_integer_field_class(self
, field_value_range
=None, preferred_display_base
=None):
250 return self
._create
_integer
_field
_class
(native_bt
.field_class_signed_integer_create
,
251 bt2
.field_class
._SignedIntegerFieldClass
,
252 'signed integer', field_value_range
, preferred_display_base
)
254 def create_unsigned_integer_field_class(self
, field_value_range
=None, preferred_display_base
=None):
255 return self
._create
_integer
_field
_class
(native_bt
.field_class_unsigned_integer_create
,
256 bt2
.field_class
._UnsignedIntegerFieldClass
,
257 'unsigned integer', field_value_range
, preferred_display_base
)
259 def create_signed_enumeration_field_class(self
, field_value_range
=None, preferred_display_base
=None):
260 return self
._create
_integer
_field
_class
(native_bt
.field_class_signed_enumeration_create
,
261 bt2
.field_class
._SignedEnumerationFieldClass
,
262 'signed enumeration', field_value_range
, preferred_display_base
)
264 def create_unsigned_enumeration_field_class(self
, field_value_range
=None, preferred_display_base
=None):
265 return self
._create
_integer
_field
_class
(native_bt
.field_class_unsigned_enumeration_create
,
266 bt2
.field_class
._UnsignedEnumerationFieldClass
,
267 'unsigned enumeration', field_value_range
, preferred_display_base
)
269 def create_real_field_class(self
, is_single_precision
=False):
270 field_class_ptr
= native_bt
.field_class_real_create(self
._ptr
)
271 self
._check
_create
_status
(field_class_ptr
, 'real')
273 field_class
= bt2
.field_class
._RealFieldClass
._create
_from
_ptr
(field_class_ptr
)
275 field_class
._is
_single
_precision
= is_single_precision
279 def create_structure_field_class(self
):
280 field_class_ptr
= native_bt
.field_class_structure_create(self
._ptr
)
281 self
._check
_create
_status
(field_class_ptr
, 'structure')
283 return bt2
.field_class
._StructureFieldClass
._create
_from
_ptr
(field_class_ptr
)
285 def create_string_field_class(self
):
286 field_class_ptr
= native_bt
.field_class_string_create(self
._ptr
)
287 self
._check
_create
_status
(field_class_ptr
, 'string')
289 return bt2
.field_class
._StringFieldClass
._create
_from
_ptr
(field_class_ptr
)
291 def create_static_array_field_class(self
, elem_fc
, length
):
292 utils
._check
_type
(elem_fc
, bt2
.field_class
._FieldClass
)
293 utils
._check
_uint
64(length
)
294 ptr
= native_bt
.field_class_static_array_create(self
._ptr
, elem_fc
._ptr
, length
)
295 self
._check
_create
_status
(ptr
, 'static array')
297 return bt2
.field_class
._StaticArrayFieldClass
._create
_from
_ptr
_and
_get
_ref
(ptr
)
299 def create_dynamic_array_field_class(self
, elem_fc
, length_fc
=None):
300 utils
._check
_type
(elem_fc
, bt2
.field_class
._FieldClass
)
301 ptr
= native_bt
.field_class_dynamic_array_create(self
._ptr
, elem_fc
._ptr
)
302 self
._check
_create
_status
(ptr
, 'dynamic array')
303 obj
= bt2
.field_class
._DynamicArrayFieldClass
._create
_from
_ptr
(ptr
)
305 if length_fc
is not None:
306 obj
._length
_field
_class
= length_fc
310 def create_variant_field_class(self
, selector_fc
=None):
311 ptr
= native_bt
.field_class_variant_create(self
._ptr
)
312 self
._check
_create
_status
(ptr
, 'variant')
313 obj
= bt2
.field_class
._VariantFieldClass
._create
_from
_ptr
(ptr
)
315 if selector_fc
is not None:
316 obj
._selector
_field
_class
= selector_fc
320 # Add a listener to be called when the trace class is destroyed.
322 def add_destruction_listener(self
, listener
):
324 if not callable(listener
):
325 raise TypeError("'listener' parameter is not callable")
327 fn
= native_bt
.py3_trace_class_add_destruction_listener
328 listener_from_native
= functools
.partial(_trace_class_destruction_listener_from_native
,
331 listener_id
= fn(self
._ptr
, listener_from_native
)
332 if listener_id
is None:
333 utils
._raise
_bt
2_error
('cannot add destruction listener to trace class object')
335 return bt2
._ListenerHandle(listener_id
, self
)