forked from legonigel/l5x
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathtag.py
759 lines (620 loc) · 29.2 KB
/
tag.py
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
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
"""
Objects implementing tag access.
"""
from .dom import (ElementAccess, ElementDict, AttributeDescriptor,
ElementDescription, CDATAElement)
import ctypes, re, struct
class Scope(ElementAccess):
"""Container to hold a group of tags within a specific scope."""
def __init__(self, element):
ElementAccess.__init__(self, element)
self.tag_element = self.get_child_element('Tags')
self.tags = ElementDict(self.tag_element, key_attr='Name', types=Tag)
class TagDataDescriptor(object):
"""Descriptor class to dispatch attribute access to a data object.
Used by Tag objects to pass access to a specific attribute on to the
Data element which handles the implementation.
"""
def __init__(self, attr):
self.attr = attr
def __get__(self, tag, owner=None):
return getattr(tag.data, self.attr)
def __set__(self, tag, value):
setattr(tag.data, self.attr, value)
class ConsumeDescriptor(object):
"""Descriptor class for accessing consumed tag properties."""
def __init__(self, attr):
self.attr = attr
def __get__(self, tag, owner=None):
if self.is_consumed(tag):
info = self.get_info(tag)
return str(info.getAttribute(self.attr))
else:
raise TypeError('Not a consumed tag')
def __set__(self, tag, value):
if not self.is_consumed(tag):
raise TypeError('Not a consumed tag')
# Producer names must be non-empty strings.
if not isinstance(value, str):
raise TypeError('Producer must be a string')
if len(value) == 0:
raise ValueError('Producer string cannot be empty')
info = self.get_info(tag)
info.setAttribute(self.attr, value)
def is_consumed(self, tag):
"""Checks to see if this is a consumed tag."""
return tag.element.getAttribute('TagType') == 'Consumed'
def get_info(self, tag):
"""Retrieves the ConsumeInfo XML element."""
return tag.get_child_element('ConsumeInfo')
class Tag(ElementAccess):
"""Base class for a single tag."""
description = ElementDescription(['ConsumeInfo'])
tag_type = AttributeDescriptor('TagType', True)
data_type = AttributeDescriptor('DataType', True)
alias_for = AttributeDescriptor('AliasFor')
value = TagDataDescriptor('value')
shape = TagDataDescriptor('shape')
names = TagDataDescriptor('names')
producer = ConsumeDescriptor('Producer')
remote_tag = ConsumeDescriptor('RemoteTag')
external_access = AttributeDescriptor('ExternalAccess', True)
constant = AttributeDescriptor('Constant', True)
def __init__(self, element):
ElementAccess.__init__(self, element)
if self.tag_type == 'Base':
data_class = base_data_types.get(self.data_type, Structure)
data_element = self.get_data_element()
if data_element is not None:
self.data = data_class(self.get_data_element(), self)
else:
self.data = None
else:
self.data = None
def get_data_element(self):
"""Returns the decorated data XML element.
This is always the sole element contained with the decorated Data
element.
Returns None if no data found
"""
if not self.tag_type == 'Base':
raise ValueError("Cannot get data element on non-base tags")
for e in self.child_elements:
if ((e.tagName == 'Data')
and (e.getAttribute('Format') == 'Decorated')):
return ElementAccess(e).child_elements[0]
return None #None if no data element
def __getitem__(self, key):
"""
Indices are passed to the data object to access members of compound
data types.
"""
if not self.tag_type == 'Base':
raise ValueError("Cannot get data on non-base tags")
if self.data is None:
raise ValueError("Cannot get data on tag with no data")
return self.data[key]
def __len__(self):
"""Dispatches len queries to the base data type object."""
if not self.tag_type == 'Base':
raise ValueError("Cannot get data on non-base tags")
if self.data is None:
raise ValueError("Cannot get data on tag with no data")
return len(self.data)
def clear_raw_data(self):
"""Removes the unformatted data element.
Called anytime a data value is set to avoid conflicts with
modified decorated data elements.
"""
if not self.tag_type == 'Base':
raise ValueError("Cannot set data on non-base tags")
for e in self.child_elements:
if (e.tagName == 'Data') and (not e.hasAttribute('Format')):
data = self.element.removeChild(e)
data.unlink()
break
@classmethod
def create(cls, scope, project, tagtype, tagname, datatype=None, value=None, description="", radix=None, dimensions="", alias_for=""):
"""
Create a tag within scope
:param scope: the scope to create the tag within (Controller, Program)
:param project: the project within which this tag is created (used for UDT definitions)
:param tagtype: The type of tag ("Base", "Alias")
:param tagname: String of the name of the tag
:param datatype: String of the datatype of the tag (Base tags only)
:param value: Value of the tag (type depends on tagtype). Arrays are lists, Structures are dicts, all base values are integers (I think) (Base tags only)
:param description: Strind description of the tag (optional)
:param radix: String of the radix type (see 1756-RM084V-EN-P) (optional) (Base tags only)
:param dimensions: String of the dimensions of the array. Currently only 1D arrays are supported (optional)
:param alias_for: String of tag this tag is an alias for (Alias only)
"""
"""Selects the Tags element to add the rung to"""
tag_element = scope.tag_element
if tagtype == "Base":
if radix is None:
radix = "Decimal" #Default to decimal radix for base tags
attributes = {'Name' : tagname,
'TagType' : tagtype,
'DataType' : datatype,
'Constant' : 'false',
'ExternalAccess' : 'Read/Write'}
is_base_datatype = datatype in base_data_types
if is_base_datatype:
attributes['Radix'] = radix
if dimensions:
#If this is an array
attributes['Dimensions'] = dimensions
element = scope._create_append_element(tag_element, 'Tag', attributes)
if value is not None: #Only make data element if value is set
data = scope._create_append_element(element, 'Data', {'Format' : 'Decorated'})
if is_base_datatype and not dimensions:
# Single base data
scope._create_append_element(data, 'DataValue', {'DataType' : datatype, \
'Radix' : radix, \
'Value' : str(value)})
elif is_base_datatype and dimensions:
# Array of base data
array = scope._create_append_element(data, 'Array',
{ 'DataType' : datatype,
'Dimensions' : dimensions,
'Radix' : radix })
for i in range(int(dimensions)):
scope._create_append_element(array, 'Element',
{'Index' : "[{}]".format(i),
'Value' : str(value[i])})
elif not dimensions:
# Single structure
Structure.create_element(scope, project, data, datatype, value)
pass
else:
# Array of Structures
array = scope._create_append_element(data, 'Array',
{ 'DataType' : datatype,
'Dimensions' : dimensions})
for i in range(int(dimensions)):
index = scope._create_append_element(array, 'Element',
{'Index' : "[{}]".format(i)})
Structure.create_element(scope, project, data, datatype, value[i])
elif tagtype == "Alias":
attributes = {'Name' : tagname,
'TagType' : tagtype,
'AliasFor' : alias_for,
'ExternalAccess' : 'Read/Write'}
if radix is not None:
attributes['Radix'] = radix
element = scope._create_append_element(tag_element, 'Tag', attributes)
else:
raise ValueError("Bad tag type {}".format(tagtype))
tag = Tag(element)
tag.description = description
scope.tags.append(tagname, tag.element)
return tag
class Comment(object):
"""Descriptor class for accessing descriptions of individual tag members.
These descriptions are stored in the Comments element directly under
the enclosing Tag element. The instance's operand attribute is used
to find the correct Comment element.
"""
def __get__(self, instance, owner=None):
"""Returns the data's description."""
try:
comments = self.get_comments(instance)
except AttributeError:
return None
try:
element = self.get_comment_element(instance, comments)
except KeyError:
return None
return str(CDATAElement(element))
def __set__(self, instance, value):
"""Updates, creates, or removes a comment."""
# Get the enclosing Comments element, creating one if necessary.
try:
comments = self.get_comments(instance)
except AttributeError:
comments = self.create_comments(instance)
# Find the matching Comment element and set the new text
# or create a new Comment if none exists.
try:
element = self.get_comment_element(instance, comments)
except KeyError:
cdata = CDATAElement(parent=comments, name='Comment',
attributes={'Operand':instance.operand})
comments.element.appendChild(cdata.element)
else:
cdata = CDATAElement(element)
if value is not None:
cdata.set(value)
else:
comments.element.removeChild(cdata.element)
def get_comments(self, instance):
"""Acquires an access object for the tag's Comments element."""
try:
element = instance.tag.get_child_element('Comments')
except KeyError:
raise AttributeError()
return ElementAccess(element)
def create_comments(self, instance):
"""Creates a new Comments container element.
Used if the top-level tag element did not contain a Comments element.
The Comments element must be located immediately before any Data
elements.
"""
new = instance.create_element('Comments')
data = instance.tag.get_child_element('Data')
instance.tag.element.insertBefore(new, data)
return ElementAccess(new)
def get_comment_element(self, instance, comments):
"""Acquires the Comment element of the instance's operand."""
for element in comments.child_elements:
if element.getAttribute('Operand') == instance.operand:
return element
raise KeyError()
class Data(ElementAccess):
"""Base class for objects providing access to tag values and comments."""
description = Comment()
# XML attribute names that contain the string used to build the operand.
# The type of attribute also determines the separator character used
# to concentate the operand with the parent's: array indices use nothing
# and everything else uses a dot.
operand_attributes = {'Name':'.', 'Index':''}
def __new__(cls, *args, **kwds):
"""
Intercepts creation of a new data object if the XML element
indicates it is an array, in which case an array access object
is created instead for the given data type.
"""
if args[0].tagName.startswith('Array'):
# Two array accessor types are possible depending on if the
# the array is a structure member.
if args[0].tagName == ('ArrayMember'):
array_type = ArrayMember
else:
array_type = Array
array = object.__new__(array_type)
array_args = [cls]
array_args.extend(args)
array.__init__(*array_args, **kwds)
return array
# Non-array tags return a instance of the original type; an explicit
# call to __init__ is not required as the returned instance
# is the original class.
else:
return object.__new__(cls)
def __init__(self, element, tag, parent=None):
ElementAccess.__init__(self, element)
self.tag = tag
self.parent = parent
self.build_operand()
def build_operand(self):
"""Constructs the identifier for comment operands.
A tag's top-level data type has no parent and does not require
an operand; it's description is placed in the dedicated Description
element. These objects get an empty operand string for child
members to use.
Operands for sub-members are formed by appending their name
to their parent's operand. Names are converted to upper-case
because Logix uses only capital letters in operand attributes
for some reason.
"""
if self.parent is None:
self.operand = ''
else:
for attr in self.operand_attributes.keys():
if self.element.hasAttribute(attr):
sep = self.operand_attributes[attr]
name = self.element.getAttribute(attr).upper()
break
self.operand = sep.join((self.parent.operand, name))
class IntegerValue(object):
"""Descriptor class for accessing an integer's value."""
def __get__(self, instance, owner=None):
if instance.element.getAttribute('Radix') == 'ASCII':
value_string = instance.element.getAttribute('Value')
value_string = value_string.replace("'","")
def logix_string_repl(matchobj):
return chr(int(matchobj.group(1)))
bytes = re.sub(r'\$(\d\d)', logix_string_repl, value_string)
bytes = bytes.rjust(4, chr(0))
return struct.unpack(">i", bytes)[0]
return int(instance.element.getAttribute('Value'))
def __set__(self, instance, value):
"""Sets a new value."""
if not isinstance(value, int):
raise TypeError('Value must be an integer')
if (value < instance.value_min) or (value > instance.value_max):
raise ValueError('Value out of range')
instance.element.setAttribute('Value', str(value))
instance.tag.clear_raw_data()
class Integer(Data):
"""Base class for integer data types.
In addition to the usual value and description access, integer indices
are used for bit-level references.
"""
value = IntegerValue()
def __getitem__(self, bit):
"""Gets an object to access a single bit."""
self.validate_bit_number(bit)
return Bit(self.element, self.tag, self, bit)
def validate_bit_number(self, bit):
"""Verifies a given bit index is within range."""
if not isinstance(bit, int):
raise TypeError('Bit indices must be integers.')
if (bit < 0) or (bit >= self.bits):
raise IndexError('Bit index out of range')
def __len__(self):
"""Returns the width of the integer."""
return self.bits
class SINT(Integer):
"""Base class for 8-bit signed integers."""
bits = 8
ctype = ctypes.c_int8
value_min = -128
value_max = 127
class INT(Integer):
"""Base class for 16-bit signed integers."""
bits = 16
ctype = ctypes.c_int16
value_min = -32768
value_max = 32767
class DINT(Integer):
"""Base class for 32-bit signed integers."""
bits = 32
ctype = ctypes.c_int32
value_min = -2147483648
value_max = 2147483647
class BitValue(object):
"""Descriptor class for values of individual integer bits.
Bit access utilizes exact-width, signed ctype integers for
bit-level operations which are then translated back to the parent
integer's value. This ensures correct results when the sign bit
is accessed.
"""
def __get__(self, bit, owner=None):
cvalue = self.get_ctype(bit)
if cvalue.value & bit.mask.value:
return 1
else:
return 0
def __set__(self, bit, bit_value):
if not isinstance(bit_value, int):
raise TypeError('Bit values must be integers')
elif (bit_value < 0) or (bit_value > 1):
raise ValueError('Bit values may only be 0 or 1')
cvalue = self.get_ctype(bit)
if bit_value:
cvalue.value |= bit.mask.value
else:
cvalue.value &= ~bit.mask.value
bit.parent.value = int(cvalue.value)
def get_ctype(self, bit):
"""Returns the parent integer's value as a ctype."""
return bit.parent.ctype(bit.parent.value)
class Bit(Data):
"""Provides access to individual bits within an integer."""
value = BitValue()
description = Comment()
def __init__(self, element, tag, parent, bit):
self.bit = bit
Data.__init__(self, element, tag, parent)
self.mask = parent.ctype(1 << bit)
def build_operand(self):
"""Method override to create an operand based on the bit number."""
self.operand = '.'.join((self.parent.operand, str(self.bit)))
class BOOL(Data):
"""Tag access for BOOL data types."""
value = IntegerValue()
value_min = 0
value_max = 1
class RealValue(object):
"""Descriptor class for accessing REAL values."""
def __get__(self, instance, owner=None):
return float(instance.element.getAttribute('Value'))
def __set__(self, instance, value):
if not isinstance(value, float):
raise TypeError('Value must be a float')
# Check for NaN and infinite values.
try:
value.as_integer_ratio()
except (OverflowError, ValueError):
raise ValueError('NaN and infinite values are not supported')
instance.element.setAttribute('Value', str(value))
instance.tag.clear_raw_data()
class REAL(Data):
"""Tag access for REAL data types."""
value = RealValue()
class StructureValue(object):
"""Descriptor class for accessing multiple structure values.
Values are expressed as a dictionary with member names as keys.
"""
def __get__(self, struct, owner=None):
member_names = struct.members.names
return dict(zip(member_names, [struct[m].value for m in member_names]))
def __set__(self, struct, value):
if not isinstance(value, dict):
raise TypeError('Value must be a dictionary')
for m in value.keys():
struct[m].value = value[m]
struct.tag.clear_raw_data()
class StructureNames(object):
"""Descriptor class for accessing structure member names."""
def __get__(self, struct, owner=None):
return struct.members.names
def __set__(self, struct, owner=None):
raise AttributeError('Read-only attribute.')
class Structure(Data):
"""Accessor class for structured data types."""
value = StructureValue()
names = StructureNames()
def __init__(self, element, tag, parent=None):
Data.__init__(self, element, tag, parent)
# If this structure is an array member the given XML element
# is just the enclosing array member; the XML element directly
# holding the structure's data is the first child: a Structure
# XML element.
if element.tagName == 'Element':
self.element = self.get_child_element('Structure')
self.members = ElementDict(self.element, key_attr='Name', types=base_data_types,
type_attr='DataType', dfl_type=Structure,
member_args=[tag, self])
def __getitem__(self, member):
"""Indexing a structure yields an individual member."""
if not isinstance(member, str):
raise TypeError('Structure indices must be strings')
return self.members[member]
@classmethod
def create_element(cls, scope, project, parent, datatype, value):
"""
Create structure data element based on type and value
:param scope: the scope to create the element within
:param project: the controller, used for UDT lookup
:param parent: parent element to this element
:param datatype: datatype of this structure
:param value: dictionary of values to put in structure
"""
if datatype not in project.datatypes:
print parent.tagName
print parent.getAttribute('Name')
raise ValueError("Datatype {} not found in datatypes".format(datatype))
else:
project_datatype = project.datatypes[datatype]
if not parent.tagName == 'StructureMember':
structure = scope._create_append_element(parent, 'Structure', {'DataType' : datatype})
else:
structure = parent
datatype_members = project_datatype.members
for i in range(len(datatype_members)):
member = datatype_members[str(i)]
if member.hidden == 'true':
# Skip hidden members
continue
member_data_type = member.data_type
if member_data_type == 'BIT':
member_data_type = 'BOOL'
if int(member.dimension):
#This member is an array
attributes = {'Name' : member.name,
'DataType' : member_data_type,
'Dimensions': member.dimension}
if member_data_type in base_data_types and member.radix:
attributes['Radix'] = member.radix
if member_data_type in base_data_types:
default_value = [0] * int(member.dimension)
else:
default_value = None
if value is not None:
data = value.get(member.name, default_value)
else:
data = default_value
array_member = scope._create_append_element(structure, 'ArrayMember', attributes)
for j in range(int(member.dimension)):
if member_data_type in base_data_types:
#Base Data Type
scope._create_append_element(array_member, 'Element', {'Index':'[{}]'.format(j), 'Value':data[j]})
else:
#Structure data type
array_element = scope._create_append_element(array_member, 'Element', {'Index':'[{}]'.format(j)})
Structure.create_element(scope, project, array_element, member_data_type, data)
else:
if value is None:
value = {}
# Not an array member
if member_data_type in base_data_types:
#Base Data Type
attributes = {'Name' : member.name,
'DataType' : member_data_type,
'Radix' : member.radix,
'Value' : value.get(member.name, 0)}
data_member = scope._create_append_element(structure, 'DataValueMember', attributes)
else:
#Structure data type
attributes = {'Name' : member.name,
'DataType' : member_data_type}
structure_member = scope._create_append_element(structure, 'StructureMember', attributes)
Structure.create_element(scope, project, structure_member, member_data_type, value.get(member.name, None))
class ArrayValue(object):
"""Descriptor class for accessing multiple values in an array."""
def __get__(self, array, owner=None):
dim = len(array.dims) - len(array.address) - 1
return [array[i].value for i in range(array.dims[dim])]
def __set__(self, array, value):
if not isinstance(value, list):
raise TypeError('Value must be a list')
if len(value) > array.shape[len(array.shape) - len(array.address) - 1]:
raise IndexError('Source list is too large')
for i in range(len(value)):
array[i].value = value[i]
array.tag.clear_raw_data()
class ArrayDescription(Comment):
"""Descriptor class array descriptions.
Raises an exception for an attempts to access descriptions because
RSLogix does not support commenting subarrays; only individual elements
may have descriptions.
"""
e = TypeError
msg = 'Descriptions for subarrays are not supported'
def __get__(self, array, owner=None):
raise self.e(self.msg)
def __set__(self, array, value):
raise self.e(self.msg)
class ArrayShape(object):
"""Descriptor class to acquire an array's dimensions."""
def __get__(self, array, owner=None):
return tuple(array.dims)
def __set__(self, array, owner=None):
raise AttributeError('Read-only attribute.')
class Array(Data):
"""Access object for arrays of any data type."""
value = ArrayValue()
description = ArrayDescription()
shape = ArrayShape()
def __init__(self, data_class, element, tag, parent=None, address=[]):
Data.__init__(self, element, tag, parent)
self.data_class = data_class
self.dims = [int(d) for d in
element.getAttribute('Dimensions').split(',')]
self.dims.reverse()
self.address = address
self.members = ElementDict(element, key_attr='Index', types=data_class,
member_args=[tag, self])
def __getitem__(self, index):
"""Returns an access object for the given index.
Multidimensional arrays will return new Array objects with the
accumulated address until all dimensions are satisfied, which
will then return the data access object for that item.
"""
if not isinstance(index, int):
raise TypeError('Array indices must be integers')
# Add the given index to the current accumulated address.
dim = len(self.dims) - len(self.address) - 1
if (index < 0) or (index >= self.dims[dim]):
raise IndexError('Array index out of range')
new_address = list(self.address)
new_address.insert(0, index)
# If the newly formed address set satisifies all dimensions
# return an access object for the member.
if len(new_address) == len(self.dims):
# Address values are reversed because the display order is
# most-significant first.
new_address.reverse()
key = "[{0}]".format(','.join([str(i) for i in new_address]))
return self.members[key]
# The new address does not yet specify a single element if the key
# was not found. Return a new array access object to handle
# access to the new address by instantiating the data type,
# which will result in an Array instance through Data.__new__().
else:
return self.data_class(self.element, self.tag, self.parent,
new_address)
class ArrayMember(Array):
"""Access object for arrays which are structure members.
Permits access to a description for the entire member. Preventing
comments for subarrays is unnecessary as array members may only be
one-dimensional.
"""
description = Comment()
base_data_types = {'SINT':SINT,
'INT':INT,
'DINT':DINT,
'BOOL':BOOL,
'REAL':REAL}