-
Notifications
You must be signed in to change notification settings - Fork 44
/
Copy pathcompiled.py
184 lines (151 loc) · 7.52 KB
/
compiled.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
import ctypes
from typing import Any, Callable, Optional
import numpy.ctypeslib as ctl
from opensourceleg.logging.logger import LOGGER
class CompiledController:
"""
Controller class to handle using compiled controllers.
This class expects that your function has the form: myFunction(*inputs, *outputs)
where *inputs is a pointer to an inputs structure and
*outputs is a pointer to an outputs structure.
You can define these input and output structures however you please.
See examples folder of repo for examples.
Args:
library_name (string): The name of the compiled library file, without the *.so
library_path (string): The path to the directory containing the library.
See examples for how to get working directory of parent script.
main_function_name (string): Name of the main function to call within the library.
This is the function that will get called via the run() method initialization_function_name (string): Name
of an initialization function for your library. This gets called only once when the library is loaded. If
you don't have an initialization function, pass None.
cleanup_function_name (string): Name of a cleanup function for your library.
This gets called when the CompiledController class has gone out of scope and is garbage collected.
Again, pass None if you don't need this functionality.
Authors:
Kevin Best, Senthur Raj Ayyappan
Neurobionics Lab
Robotics Department
University of Michigan
October 2023
"""
def __init__(
self,
library_name: str,
library_path: str,
main_function_name: str,
initialization_function_name: Optional[str] = None,
cleanup_function_name: Optional[str] = None,
) -> None:
self.lib: Any = ctl.load_library(library_name, library_path)
self.cleanup_func: Callable = self._load_function(cleanup_function_name)
self.main_function: Callable = self._load_function(main_function_name)
self.init_function: Callable = self._load_function(initialization_function_name)
# Note if requested function name is None, returned handle is also none
if self.init_function is not None:
self.init_function()
# This alias makes defining types from top script easier without second import
self.types = ctypes
self.DEFAULT_SENSOR_LIST = [
("knee_angle", self.types.c_double),
("ankle_angle", self.types.c_double),
("knee_velocity", self.types.c_double),
("ankle_velocity", self.types.c_double),
("Fz", self.types.c_double),
]
self._input_type = None
self.inputs = None
self._output_type = None
self.outputs = None
def __del__(self) -> None:
if hasattr(self, "cleanup_func") and self.cleanup_func is not None:
self.cleanup_func()
def __repr__(self) -> str:
return "CompiledController"
def _load_function(self, function_name: Optional[str]) -> Any:
if function_name is None:
return None
else:
try:
function_handle = getattr(self.lib, function_name)
except AttributeError:
LOGGER.warning(f"Function {function_name} not found in library {self.lib}")
return function_handle
def define_inputs(self, input_list: list[Any]) -> None:
"""
This method defines the input structure to your function.
See example folder and tutorials for help on using this method.
Parameters
-----------
input_list: Input parameters given as a list of [('field_name', field_type)...]
field_name is a string you choose as the title of the field.
field_type is a type either given by a native c_types value or
a custom type defined via the define_type() method.
All types can be accessed as CompiledController.types.(type_name)
"""
self._input_type = self.define_type("inputs", input_list)
if self._input_type is None:
raise ValueError("Input type not defined properly. Check define_type() method.")
self.inputs = self._input_type()
def define_outputs(self, output_list: list[Any]) -> None:
"""
This method defines the output structure to your function.
See example folder and tutorials for help on using this method.
Parameters
------------
output_list: Output parameters given as a list of [('field_name', field_type)...]
field_name is a string you choose as the title of the field.
field_type is a type either given by a native c_types value or
a custom type defined via the define_type() method.
All types can be accessed as CompiledController.types.(type_name)
"""
self._output_type = self.define_type("outputs", output_list)
if self._output_type is None:
raise ValueError("Output type not defined properly. Check define_type() method.")
self.outputs = self._output_type()
def define_type(self, type_name: str, parameter_list: list[Any]) -> Any:
"""
This method defines a new type to be used in the compiled controller.
After calling this method, the datatype with name type_name will be
available in my_controller.types.type_name for use.
See example folder and tutorials for help on using this method.
Parameters
------------
type_name : A string defining the name of your new datatype
parameter_list: A list of [('field_name', field_type)...]
field_name is a string you choose as the title of the field.
field_type is a type either given by a native c_types value or
a custom type defined via the define_type() method.
All types can be accessed as CompiledController.types.(type_name)
Example Usage
------------
my_controller.DefineType('vector3D', [('x', my_controller.types.c_double),
('y', my_controller.types.c_double),
('z', my_controller.types.c_double)])
"""
slots = []
for param in parameter_list:
slots.append(param[0])
class CustomStructure(ctypes.Structure):
_fields_ = parameter_list
__slots__ = slots
setattr(self.types, type_name, CustomStructure)
return getattr(self.types, type_name)
def run(self) -> Any:
"""
This method calls the main controller function of the library.
Under the hood, it calls library_name.main_function_name(*inputs, *outputs),
where library_name and main_function_name were given in the constructor.
Parameters -> None
Returns:
The output structure as defined by the define_outputs() method.
Raises:
ValueError: If define_inputs() or define_outputs() have not been called.
"""
if self.inputs is None:
raise ValueError("Must define input type before calling controller.run(). Use define_inputs() method.")
if self.outputs is None:
raise ValueError("Must define output type before calling controller.run(). Use define_outputs() method.")
self.main_function(ctypes.byref(self.inputs), ctypes.byref(self.outputs))
return self.outputs
if __name__ == "__main__":
pass