Important changes to repositories hosted on mbed.com
Mbed hosted mercurial repositories are deprecated and are due to be permanently deleted in July 2026.
To keep a copy of this software download the repository Zip archive or clone locally using Mercurial.
It is also possible to export all your personal repositories from the account settings page.
Dependencies: NaturalTinyShell_ice libmDot-12Sept mbed-rtos mbed
Fork of ICE by
src/ConfigurationHandler/ConfigurationHandler.cpp@93:1553fb156915, 2016-09-20 (annotated)
- Committer:
- jmarkel44
- Date:
- Tue Sep 20 19:31:38 2016 +0000
- Revision:
- 93:1553fb156915
- Parent:
- 90:7d9731dec0da
- Child:
- 97:5cf6ab71dcd0
setpoint control cleanup;
Who changed what in which revision?
User | Revision | Line number | New contents of line |
---|---|---|---|
jmarkel44 | 0:65cfa4873284 | 1 | /****************************************************************************** |
jmarkel44 | 0:65cfa4873284 | 2 | * |
jmarkel44 | 0:65cfa4873284 | 3 | * File: ConfigurationHandler.cpp |
jmarkel44 | 0:65cfa4873284 | 4 | * Desciption: source for the ICE Configuration Handler |
jmarkel44 | 0:65cfa4873284 | 5 | * |
jmarkel44 | 0:65cfa4873284 | 6 | *****************************************************************************/ |
jmarkel44 | 3:8ea4db957749 | 7 | #include "ConfigurationHandler.h" |
jmarkel44 | 0:65cfa4873284 | 8 | #include "global.h" |
jmarkel44 | 5:5e77a1db4d45 | 9 | #include "SetpointControl.h" |
jmarkel44 | 12:ea87887ca7ad | 10 | #include "TimerControl.h" |
jmarkel44 | 12:ea87887ca7ad | 11 | |
jmarkel44 | 46:4cb96ab2d1c8 | 12 | StringSetpointMap setpointTable; // setpoint control object table |
jmarkel44 | 46:4cb96ab2d1c8 | 13 | StringTimerMap timerTable; // timer control object table |
jmarkel44 | 46:4cb96ab2d1c8 | 14 | //StringManualMap manualTable; // manual control object table |
jmarkel44 | 5:5e77a1db4d45 | 15 | |
jmarkel44 | 5:5e77a1db4d45 | 16 | // local function prototypes |
jmarkel44 | 5:5e77a1db4d45 | 17 | static int loadPersistentControls(void); |
jmarkel44 | 5:5e77a1db4d45 | 18 | |
jmarkel44 | 74:03ccf04998b5 | 19 | // local helper functions |
jmarkel44 | 5:5e77a1db4d45 | 20 | static int createControl(const Message_t *msg); |
jmarkel44 | 5:5e77a1db4d45 | 21 | static int modifyControl(const Message_t *msg); |
jmarkel44 | 5:5e77a1db4d45 | 22 | static int destroyControl(const Message_t *msg); |
jmarkel44 | 5:5e77a1db4d45 | 23 | |
jmarkel44 | 12:ea87887ca7ad | 24 | /***************************************************************************** |
jmarkel44 | 12:ea87887ca7ad | 25 | * Function: ConfigurationHandler() |
jmarkel44 | 12:ea87887ca7ad | 26 | * Description: The ICE Configuration Handler |
jmarkel44 | 12:ea87887ca7ad | 27 | * |
jmarkel44 | 12:ea87887ca7ad | 28 | * @param args (unused) |
jmarkel44 | 12:ea87887ca7ad | 29 | * @return none |
jmarkel44 | 12:ea87887ca7ad | 30 | *****************************************************************************/ |
jmarkel44 | 0:65cfa4873284 | 31 | void ConfigurationHandler(void const *args) |
jmarkel44 | 0:65cfa4873284 | 32 | { |
jmarkel44 | 12:ea87887ca7ad | 33 | UNUSED(args); |
jmarkel44 | 5:5e77a1db4d45 | 34 | loadPersistentControls(); |
jmarkel44 | 75:96512ccc0443 | 35 | osSignalSet(mainThreadId, sig_config_continue); |
jmarkel44 | 5:5e77a1db4d45 | 36 | |
jmarkel44 | 3:8ea4db957749 | 37 | while ( true ) { |
jmarkel44 | 5:5e77a1db4d45 | 38 | // wait for an event |
jmarkel44 | 5:5e77a1db4d45 | 39 | osEvent evt = MailBox.get(); |
jmarkel44 | 5:5e77a1db4d45 | 40 | if (evt.status == osEventMail) { |
jmarkel44 | 5:5e77a1db4d45 | 41 | Message_t *msg = (Message_t*) evt.value.p; |
jmarkel44 | 5:5e77a1db4d45 | 42 | |
jmarkel44 | 37:7e6986b77f01 | 43 | logInfo("\r%s: mes->action = %d\n", __func__, msg->action); |
jmarkel44 | 37:7e6986b77f01 | 44 | logInfo("\r%s: msg->control = %d\n", __func__, msg->control); |
jmarkel44 | 37:7e6986b77f01 | 45 | logInfo("\r%s: msg->controlFile = %s\n", __func__, msg->controlFile); |
jmarkel44 | 5:5e77a1db4d45 | 46 | |
jmarkel44 | 5:5e77a1db4d45 | 47 | switch ( msg->action ) { |
jmarkel44 | 5:5e77a1db4d45 | 48 | case ACTION_CREATE: { |
jmarkel44 | 5:5e77a1db4d45 | 49 | (void)createControl(msg); |
jmarkel44 | 5:5e77a1db4d45 | 50 | break; |
jmarkel44 | 5:5e77a1db4d45 | 51 | } |
jmarkel44 | 5:5e77a1db4d45 | 52 | case ACTION_MODIFY: { |
jmarkel44 | 5:5e77a1db4d45 | 53 | (void)modifyControl(msg); |
jmarkel44 | 5:5e77a1db4d45 | 54 | break; |
jmarkel44 | 5:5e77a1db4d45 | 55 | } |
jmarkel44 | 5:5e77a1db4d45 | 56 | case ACTION_DESTROY: { |
jmarkel44 | 5:5e77a1db4d45 | 57 | (void)destroyControl(msg); |
jmarkel44 | 5:5e77a1db4d45 | 58 | break; |
jmarkel44 | 5:5e77a1db4d45 | 59 | } |
jmarkel44 | 5:5e77a1db4d45 | 60 | default: |
jmarkel44 | 5:5e77a1db4d45 | 61 | break; |
jmarkel44 | 5:5e77a1db4d45 | 62 | } |
jmarkel44 | 5:5e77a1db4d45 | 63 | |
jmarkel44 | 5:5e77a1db4d45 | 64 | // free the message |
jmarkel44 | 5:5e77a1db4d45 | 65 | MailBox.free(msg); |
jmarkel44 | 5:5e77a1db4d45 | 66 | } |
jmarkel44 | 0:65cfa4873284 | 67 | } |
jmarkel44 | 0:65cfa4873284 | 68 | } |
jmarkel44 | 5:5e77a1db4d45 | 69 | |
jmarkel44 | 12:ea87887ca7ad | 70 | /***************************************************************************** |
jmarkel44 | 12:ea87887ca7ad | 71 | * Function: ConfigurationHandler_showControls() |
jmarkel44 | 13:c80c283f9db2 | 72 | * Description: show the controls |
jmarkel44 | 12:ea87887ca7ad | 73 | * |
jmarkel44 | 12:ea87887ca7ad | 74 | * @param msg |
jmarkel44 | 12:ea87887ca7ad | 75 | * @return none |
jmarkel44 | 12:ea87887ca7ad | 76 | *****************************************************************************/ |
jmarkel44 | 12:ea87887ca7ad | 77 | void ConfigurationHandler_showControls(void) |
jmarkel44 | 12:ea87887ca7ad | 78 | { |
jmarkel44 | 12:ea87887ca7ad | 79 | if ( !timerTable.empty() ) { |
jmarkel44 | 12:ea87887ca7ad | 80 | printf("\rTIMER CONTROLS\n"); |
jmarkel44 | 12:ea87887ca7ad | 81 | StringTimerMap::iterator pos; |
jmarkel44 | 46:4cb96ab2d1c8 | 82 | for ( pos = timerTable.begin(); pos != timerTable.end(); ++pos ) { |
jmarkel44 | 46:4cb96ab2d1c8 | 83 | printf("\r control file: %32s\n", |
jmarkel44 | 46:4cb96ab2d1c8 | 84 | pos->second->getControlFile().c_str()); |
jmarkel44 | 12:ea87887ca7ad | 85 | } |
jmarkel44 | 12:ea87887ca7ad | 86 | } |
jmarkel44 | 74:03ccf04998b5 | 87 | |
jmarkel44 | 12:ea87887ca7ad | 88 | if ( !setpointTable.empty() ) { |
jmarkel44 | 12:ea87887ca7ad | 89 | printf("\rSETPOINT CONTROLS\n"); |
jmarkel44 | 12:ea87887ca7ad | 90 | StringSetpointMap::iterator pos; |
jmarkel44 | 93:1553fb156915 | 91 | for ( pos = setpointTable.begin(); pos != setpointTable.end(); ++pos ) { |
jmarkel44 | 93:1553fb156915 | 92 | pos->second->display(); |
jmarkel44 | 12:ea87887ca7ad | 93 | } |
jmarkel44 | 12:ea87887ca7ad | 94 | } |
jmarkel44 | 12:ea87887ca7ad | 95 | } |
jmarkel44 | 12:ea87887ca7ad | 96 | |
jmarkel44 | 12:ea87887ca7ad | 97 | /***************************************************************************** |
jmarkel44 | 12:ea87887ca7ad | 98 | * Function: loadPersistentControls() |
jmarkel44 | 12:ea87887ca7ad | 99 | * Description: load persistent controls from flash |
jmarkel44 | 12:ea87887ca7ad | 100 | * |
jmarkel44 | 12:ea87887ca7ad | 101 | * @param none |
jmarkel44 | 12:ea87887ca7ad | 102 | * @return none |
jmarkel44 | 12:ea87887ca7ad | 103 | *****************************************************************************/ |
jmarkel44 | 12:ea87887ca7ad | 104 | static int loadPersistentControls(void) |
jmarkel44 | 12:ea87887ca7ad | 105 | { |
jmarkel44 | 12:ea87887ca7ad | 106 | static bool loaded = false; |
jmarkel44 | 12:ea87887ca7ad | 107 | |
jmarkel44 | 12:ea87887ca7ad | 108 | if ( !loaded ) { // lazy protection |
jmarkel44 | 77:43e0a3d9e536 | 109 | |
jmarkel44 | 77:43e0a3d9e536 | 110 | printf("\rLoading persistent controls: \n"); |
jmarkel44 | 74:03ccf04998b5 | 111 | std::vector<mDot::mdot_file> file_list = GLOBAL_mdot->listUserFiles(); |
jmarkel44 | 74:03ccf04998b5 | 112 | |
jmarkel44 | 74:03ccf04998b5 | 113 | for (std::vector<mDot::mdot_file>::iterator i = file_list.begin(); i != file_list.end(); ++i) { |
jmarkel44 | 74:03ccf04998b5 | 114 | if( strncmp( i->name, CONTROL_SP_STR, strlen(CONTROL_SP_STR)) == 0 ) { |
jmarkel44 | 77:43e0a3d9e536 | 115 | // create the setpoint control |
jmarkel44 | 74:03ccf04998b5 | 116 | Message_t msg; |
jmarkel44 | 74:03ccf04998b5 | 117 | msg.control = CONTROL_SETPOINT; |
jmarkel44 | 74:03ccf04998b5 | 118 | strncpy(msg.controlFile, i->name, sizeof(msg.controlFile)); |
jmarkel44 | 74:03ccf04998b5 | 119 | int rc = createControl(&msg); |
jmarkel44 | 74:03ccf04998b5 | 120 | if ( rc != 0 ) { |
jmarkel44 | 74:03ccf04998b5 | 121 | } else { |
jmarkel44 | 74:03ccf04998b5 | 122 | printf("\r control %s loaded.\n", msg.controlFile); |
jmarkel44 | 74:03ccf04998b5 | 123 | } |
jmarkel44 | 74:03ccf04998b5 | 124 | } else if ( strncmp( i->name, CONTROL_TM_STR, strlen(CONTROL_TM_STR)) == 0 ) { |
jmarkel44 | 77:43e0a3d9e536 | 125 | // create the timer control |
jmarkel44 | 74:03ccf04998b5 | 126 | Message_t msg; |
jmarkel44 | 74:03ccf04998b5 | 127 | msg.control = CONTROL_TIMER; |
jmarkel44 | 74:03ccf04998b5 | 128 | strncpy(msg.controlFile, i->name, sizeof(msg.controlFile)); |
jmarkel44 | 74:03ccf04998b5 | 129 | int rc = createControl(&msg); |
jmarkel44 | 74:03ccf04998b5 | 130 | if ( rc != 0 ) { |
jmarkel44 | 74:03ccf04998b5 | 131 | } else { |
jmarkel44 | 74:03ccf04998b5 | 132 | printf("\r control %s loaded.\n", msg.controlFile); |
jmarkel44 | 74:03ccf04998b5 | 133 | } |
jmarkel44 | 74:03ccf04998b5 | 134 | } else if ( strncmp( i->name, CONTROL_MN_STR, strlen(CONTROL_MN_STR)) == 0 ) { |
jmarkel44 | 77:43e0a3d9e536 | 135 | // TODO: load the manual control -> these may not persist?? |
jmarkel44 | 77:43e0a3d9e536 | 136 | // do nothing |
jmarkel44 | 74:03ccf04998b5 | 137 | } else { |
jmarkel44 | 74:03ccf04998b5 | 138 | // not a control file |
jmarkel44 | 74:03ccf04998b5 | 139 | } |
jmarkel44 | 74:03ccf04998b5 | 140 | } |
jmarkel44 | 12:ea87887ca7ad | 141 | } |
jmarkel44 | 12:ea87887ca7ad | 142 | return 0; |
jmarkel44 | 12:ea87887ca7ad | 143 | } |
jmarkel44 | 12:ea87887ca7ad | 144 | |
jmarkel44 | 12:ea87887ca7ad | 145 | /***************************************************************************** |
jmarkel44 | 12:ea87887ca7ad | 146 | * Function: createControl() |
jmarkel44 | 12:ea87887ca7ad | 147 | * Description: creates a new control |
jmarkel44 | 12:ea87887ca7ad | 148 | * |
jmarkel44 | 12:ea87887ca7ad | 149 | * @param none |
jmarkel44 | 12:ea87887ca7ad | 150 | * @return none |
jmarkel44 | 12:ea87887ca7ad | 151 | *****************************************************************************/ |
jmarkel44 | 5:5e77a1db4d45 | 152 | static int createControl(const Message_t *msg) |
jmarkel44 | 5:5e77a1db4d45 | 153 | { |
jmarkel44 | 37:7e6986b77f01 | 154 | logInfo("\r%s invoked\n", __func__); |
jmarkel44 | 5:5e77a1db4d45 | 155 | |
jmarkel44 | 5:5e77a1db4d45 | 156 | switch (msg->control) { |
jmarkel44 | 5:5e77a1db4d45 | 157 | case CONTROL_SETPOINT: { |
jmarkel44 | 46:4cb96ab2d1c8 | 158 | SetpointControl *setpointControl = new SetpointControl; |
jmarkel44 | 28:c410a61238bb | 159 | bool rc = setpointControl->load(msg->controlFile); |
jmarkel44 | 19:9bc8fabeddfa | 160 | if ( rc != true ) { |
jmarkel44 | 19:9bc8fabeddfa | 161 | logError("%s: failed to load %s\n", __func__, msg->controlFile); |
jmarkel44 | 19:9bc8fabeddfa | 162 | delete setpointControl; |
jmarkel44 | 19:9bc8fabeddfa | 163 | } else { |
jmarkel44 | 19:9bc8fabeddfa | 164 | setpointTable[msg->controlFile] = setpointControl; |
jmarkel44 | 74:03ccf04998b5 | 165 | // start the setpoint control |
jmarkel44 | 74:03ccf04998b5 | 166 | setpointControl->start(); |
jmarkel44 | 19:9bc8fabeddfa | 167 | } |
jmarkel44 | 5:5e77a1db4d45 | 168 | break; |
jmarkel44 | 5:5e77a1db4d45 | 169 | } |
jmarkel44 | 5:5e77a1db4d45 | 170 | case CONTROL_TIMER: { |
jmarkel44 | 46:4cb96ab2d1c8 | 171 | TimerControl *timerControl = new TimerControl; |
jmarkel44 | 28:c410a61238bb | 172 | bool rc = timerControl->load(msg->controlFile); |
jmarkel44 | 19:9bc8fabeddfa | 173 | if ( rc != true ) { |
jmarkel44 | 19:9bc8fabeddfa | 174 | logError("%s: failed to load %s\n", __func__, msg->controlFile); |
jmarkel44 | 19:9bc8fabeddfa | 175 | delete timerControl; |
jmarkel44 | 19:9bc8fabeddfa | 176 | } else { |
jmarkel44 | 19:9bc8fabeddfa | 177 | timerTable[msg->controlFile] = timerControl; |
jmarkel44 | 19:9bc8fabeddfa | 178 | } |
jmarkel44 | 5:5e77a1db4d45 | 179 | break; |
jmarkel44 | 5:5e77a1db4d45 | 180 | } |
jmarkel44 | 46:4cb96ab2d1c8 | 181 | case CONTROL_MANUAL: { |
jmarkel44 | 46:4cb96ab2d1c8 | 182 | #if 0 |
jmarkel44 | 46:4cb96ab2d1c8 | 183 | ManualControl *manualControl = new ManualControl; |
jmarkel44 | 46:4cb96ab2d1c8 | 184 | bool rc = manualControl->load(msg->controlFile); |
jmarkel44 | 46:4cb96ab2d1c8 | 185 | if ( rc != true ) { |
jmarkel44 | 46:4cb96ab2d1c8 | 186 | logError("%s: failed to load %s\n", __func__, msg->controlFile); |
jmarkel44 | 46:4cb96ab2d1c8 | 187 | delete manualControl; |
jmarkel44 | 46:4cb96ab2d1c8 | 188 | } else { |
jmarkel44 | 46:4cb96ab2d1c8 | 189 | manualTable[msg->controlFile] = manualControl; |
jmarkel44 | 46:4cb96ab2d1c8 | 190 | } |
jmarkel44 | 46:4cb96ab2d1c8 | 191 | #endif |
jmarkel44 | 46:4cb96ab2d1c8 | 192 | break; |
jmarkel44 | 46:4cb96ab2d1c8 | 193 | } |
jmarkel44 | 19:9bc8fabeddfa | 194 | case CONTROL_PID: |
jmarkel44 | 19:9bc8fabeddfa | 195 | case CONTROL_COMPOSITE: |
jmarkel44 | 5:5e77a1db4d45 | 196 | default: |
jmarkel44 | 46:4cb96ab2d1c8 | 197 | logInfo("\r%s: control type %d not implemented yet...\n", |
jmarkel44 | 46:4cb96ab2d1c8 | 198 | __func__, msg->control); |
jmarkel44 | 5:5e77a1db4d45 | 199 | break; |
jmarkel44 | 5:5e77a1db4d45 | 200 | } |
jmarkel44 | 5:5e77a1db4d45 | 201 | return 0; |
jmarkel44 | 5:5e77a1db4d45 | 202 | } |
jmarkel44 | 12:ea87887ca7ad | 203 | |
jmarkel44 | 12:ea87887ca7ad | 204 | /***************************************************************************** |
jmarkel44 | 12:ea87887ca7ad | 205 | * Function: modifyControl() |
jmarkel44 | 12:ea87887ca7ad | 206 | * Description: modifies a control |
jmarkel44 | 12:ea87887ca7ad | 207 | * |
jmarkel44 | 12:ea87887ca7ad | 208 | * @param msg |
jmarkel44 | 12:ea87887ca7ad | 209 | * @return none |
jmarkel44 | 12:ea87887ca7ad | 210 | *****************************************************************************/ |
jmarkel44 | 5:5e77a1db4d45 | 211 | static int modifyControl(const Message_t *msg) |
jmarkel44 | 5:5e77a1db4d45 | 212 | { |
jmarkel44 | 37:7e6986b77f01 | 213 | logInfo("\r%s invoked\n", __func__); |
jmarkel44 | 46:4cb96ab2d1c8 | 214 | |
jmarkel44 | 46:4cb96ab2d1c8 | 215 | // TODO: can we delete the current object and start a new one? |
jmarkel44 | 5:5e77a1db4d45 | 216 | return 0; |
jmarkel44 | 5:5e77a1db4d45 | 217 | |
jmarkel44 | 5:5e77a1db4d45 | 218 | } |
jmarkel44 | 12:ea87887ca7ad | 219 | |
jmarkel44 | 12:ea87887ca7ad | 220 | /***************************************************************************** |
jmarkel44 | 12:ea87887ca7ad | 221 | * Function: destroyControl() |
jmarkel44 | 12:ea87887ca7ad | 222 | * Description: destroys a controls |
jmarkel44 | 12:ea87887ca7ad | 223 | * |
jmarkel44 | 12:ea87887ca7ad | 224 | * @param msg |
jmarkel44 | 12:ea87887ca7ad | 225 | * @return none |
jmarkel44 | 12:ea87887ca7ad | 226 | *****************************************************************************/ |
jmarkel44 | 5:5e77a1db4d45 | 227 | static int destroyControl(const Message_t *msg) |
jmarkel44 | 5:5e77a1db4d45 | 228 | { |
jmarkel44 | 37:7e6986b77f01 | 229 | logInfo("\r%s invoked\n", __func__); |
jmarkel44 | 12:ea87887ca7ad | 230 | |
jmarkel44 | 12:ea87887ca7ad | 231 | switch ( msg->control ) { |
jmarkel44 | 12:ea87887ca7ad | 232 | case CONTROL_SETPOINT: { |
jmarkel44 | 12:ea87887ca7ad | 233 | StringSetpointMap::iterator pos; |
jmarkel44 | 12:ea87887ca7ad | 234 | pos = setpointTable.find(msg->controlFile); |
jmarkel44 | 12:ea87887ca7ad | 235 | if ( pos != setpointTable.end() ) { |
jmarkel44 | 74:03ccf04998b5 | 236 | pos->second->unregisterControl(); |
jmarkel44 | 12:ea87887ca7ad | 237 | delete (pos->second); |
jmarkel44 | 12:ea87887ca7ad | 238 | setpointTable.erase(pos); |
jmarkel44 | 12:ea87887ca7ad | 239 | } |
jmarkel44 | 12:ea87887ca7ad | 240 | break; |
jmarkel44 | 12:ea87887ca7ad | 241 | } |
jmarkel44 | 12:ea87887ca7ad | 242 | case CONTROL_TIMER: { |
jmarkel44 | 12:ea87887ca7ad | 243 | StringTimerMap::iterator pos; |
jmarkel44 | 12:ea87887ca7ad | 244 | pos = timerTable.find(msg->controlFile); |
jmarkel44 | 12:ea87887ca7ad | 245 | if ( pos != timerTable.end() ) { |
jmarkel44 | 12:ea87887ca7ad | 246 | delete (pos->second); |
jmarkel44 | 12:ea87887ca7ad | 247 | timerTable.erase(pos); |
jmarkel44 | 12:ea87887ca7ad | 248 | } |
jmarkel44 | 12:ea87887ca7ad | 249 | break; |
jmarkel44 | 12:ea87887ca7ad | 250 | } |
jmarkel44 | 46:4cb96ab2d1c8 | 251 | case CONTROL_MANUAL: { |
jmarkel44 | 46:4cb96ab2d1c8 | 252 | #if 0 |
jmarkel44 | 46:4cb96ab2d1c8 | 253 | StringManualMap::iterator pos; |
jmarkel44 | 46:4cb96ab2d1c8 | 254 | pos = manualTable.find(msg->controlFile); |
jmarkel44 | 46:4cb96ab2d1c8 | 255 | if ( pos != manualTable.end() ) { |
jmarkel44 | 46:4cb96ab2d1c8 | 256 | delete (pos->second); |
jmarkel44 | 46:4cb96ab2d1c8 | 257 | manualTable.erase(pos); |
jmarkel44 | 46:4cb96ab2d1c8 | 258 | } |
jmarkel44 | 46:4cb96ab2d1c8 | 259 | #endif |
jmarkel44 | 46:4cb96ab2d1c8 | 260 | break; |
jmarkel44 | 46:4cb96ab2d1c8 | 261 | } |
jmarkel44 | 12:ea87887ca7ad | 262 | default: |
jmarkel44 | 12:ea87887ca7ad | 263 | break; |
jmarkel44 | 12:ea87887ca7ad | 264 | } |
jmarkel44 | 12:ea87887ca7ad | 265 | return 0; |
jmarkel44 | 20:653923c2f37a | 266 | } |