2010-10-18 15:24:46 -03:00
|
|
|
/*
|
|
|
|
www.ArduCopter.com - www.DIYDrones.com
|
|
|
|
Copyright (c) 2010. All rights reserved.
|
|
|
|
An Open Source Arduino based multicopter.
|
|
|
|
|
|
|
|
File : CLI.pde
|
|
|
|
Version : v1.0, Oct 18, 2010
|
|
|
|
Author(s): ArduCopter Team
|
|
|
|
Jani Hirvinen, Jose Julio, Jordi Muñoz,
|
2010-10-30 05:30:46 -03:00
|
|
|
Ken McEwans, Roberto Navoni, Sandro Benigno, Chris Anderson, Randy McEvans
|
2010-10-18 15:24:46 -03:00
|
|
|
|
|
|
|
This program is free software: you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
* ************************************************************** *
|
|
|
|
ChangeLog:
|
2010-10-19 12:22:58 -03:00
|
|
|
- 19-10-10 Initial CLI
|
2010-10-18 15:24:46 -03:00
|
|
|
|
|
|
|
* ************************************************************** *
|
|
|
|
TODO:
|
2010-10-19 12:22:58 -03:00
|
|
|
- Full menu systems, debug, settings
|
2010-10-18 15:24:46 -03:00
|
|
|
|
|
|
|
* ************************************************************** */
|
|
|
|
|
2010-12-08 07:59:14 -04:00
|
|
|
|
2010-10-19 12:22:58 -03:00
|
|
|
boolean ShowMainMenu;
|
|
|
|
|
|
|
|
|
2010-10-18 15:24:46 -03:00
|
|
|
// CLI Functions
|
|
|
|
// This can be moved later to CLI.pde
|
|
|
|
void RunCLI () {
|
|
|
|
|
2010-11-28 06:56:27 -04:00
|
|
|
// APM_RC.Init(); // APM Radio initialization
|
|
|
|
|
2010-10-30 05:30:46 -03:00
|
|
|
readUserConfig(); // Read memory values from EEPROM
|
2010-11-28 06:56:27 -04:00
|
|
|
|
2010-10-19 12:22:58 -03:00
|
|
|
ShowMainMenu = TRUE;
|
2010-10-18 15:24:46 -03:00
|
|
|
// We need to initialize Serial again due it was not initialized during startup.
|
|
|
|
SerBeg(SerBau);
|
2010-10-19 12:22:58 -03:00
|
|
|
SerPrln();
|
|
|
|
SerPrln("Welcome to ArduCopter CLI");
|
2010-10-18 15:24:46 -03:00
|
|
|
SerPri("Firmware: ");
|
2010-10-19 12:22:58 -03:00
|
|
|
SerPrln(VER);
|
|
|
|
SerPrln();
|
2010-12-08 08:19:21 -04:00
|
|
|
SerPrln("Make sure that you have Carriage Return active");
|
2010-10-19 12:22:58 -03:00
|
|
|
|
|
|
|
if(ShowMainMenu) Show_MainMenu();
|
|
|
|
|
2010-10-18 15:24:46 -03:00
|
|
|
// Our main loop that never ends. Only way to get away from here is to reboot APM
|
|
|
|
for (;;) {
|
2010-12-08 07:59:14 -04:00
|
|
|
|
|
|
|
if(ShowMainMenu) Show_MainMenu();
|
|
|
|
|
|
|
|
delay(50);
|
2010-10-19 12:22:58 -03:00
|
|
|
if (SerAva()) {
|
|
|
|
ShowMainMenu = TRUE;
|
|
|
|
queryType = SerRea();
|
|
|
|
switch (queryType) {
|
|
|
|
case 'c':
|
|
|
|
CALIB_CompassOffset();
|
|
|
|
break;
|
2010-10-30 05:30:46 -03:00
|
|
|
case 'i':
|
|
|
|
CALIB_AccOffset();
|
2010-12-08 07:59:14 -04:00
|
|
|
break;
|
|
|
|
case 't':
|
|
|
|
CALIB_Throttle();
|
|
|
|
break;
|
|
|
|
case 'e':
|
|
|
|
CALIB_Esc();
|
2010-10-30 05:30:46 -03:00
|
|
|
break;
|
2010-12-08 07:59:14 -04:00
|
|
|
case 's':
|
|
|
|
Show_Settings();
|
|
|
|
break;
|
2010-10-19 12:22:58 -03:00
|
|
|
}
|
2010-12-08 07:59:14 -04:00
|
|
|
SerFlu();
|
2010-10-19 12:22:58 -03:00
|
|
|
}
|
2010-12-08 07:59:14 -04:00
|
|
|
|
2010-11-28 06:56:27 -04:00
|
|
|
// Changing LED statuses to inform that we are in CLI mode
|
|
|
|
// Blinking Red, Yellow, Green when in CLI mode
|
|
|
|
if(millis() - cli_timer > 1000) {
|
|
|
|
cli_timer = millis();
|
2010-11-28 07:16:38 -04:00
|
|
|
CLILedStep();
|
2010-11-28 06:56:27 -04:00
|
|
|
}
|
2010-12-08 07:59:14 -04:00
|
|
|
} // Mainloop ends
|
|
|
|
}
|
2010-11-28 06:56:27 -04:00
|
|
|
|
|
|
|
|
2010-10-18 15:24:46 -03:00
|
|
|
|
|
|
|
|
2010-10-19 12:22:58 -03:00
|
|
|
void Show_MainMenu() {
|
|
|
|
ShowMainMenu = FALSE;
|
2010-12-08 07:59:14 -04:00
|
|
|
SerPrln();
|
2010-10-19 12:22:58 -03:00
|
|
|
SerPrln("CLI Menu - Type your command on command prompt");
|
|
|
|
SerPrln("----------------------------------------------");
|
2010-12-08 07:59:14 -04:00
|
|
|
SerPrln(" c - Show compass offsets");
|
2010-12-08 08:19:21 -04:00
|
|
|
SerPrln(" e - ESC Throttle calibration (Works with official ArduCopter ESCs)");
|
|
|
|
SerPrln(" i - Initialize and calibrate Accel offsets");
|
2010-12-08 07:59:14 -04:00
|
|
|
SerPrln(" t - Calibrate MIN Throttle value");
|
|
|
|
SerPrln(" s - Show settings");
|
2010-10-19 12:22:58 -03:00
|
|
|
SerPrln(" ");
|
2010-12-08 07:59:14 -04:00
|
|
|
SerFlu();
|
2010-10-19 12:22:58 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* ************************************************************** */
|
|
|
|
// Compass/Magnetometer Offset Calibration
|
|
|
|
void CALIB_CompassOffset() {
|
|
|
|
|
|
|
|
#ifdef IsMAG
|
|
|
|
SerPrln("Rotate/Pitch/Roll your ArduCopter untill offset variables are not");
|
|
|
|
SerPrln("anymore changing, write down your offset values and update them ");
|
|
|
|
SerPrln("to their correct location. Starting in..");
|
|
|
|
SerPrln("2 secs.");
|
|
|
|
delay(1000);
|
|
|
|
SerPrln("1 secs.");
|
|
|
|
delay(1000);
|
|
|
|
SerPrln("starting now....");
|
|
|
|
|
|
|
|
APM_Compass.Init(); // Initialization
|
|
|
|
APM_Compass.SetOrientation(MAGORIENTATION); // set compass's orientation on aircraft
|
|
|
|
APM_Compass.SetOffsets(0,0,0); // set offsets to account for surrounding interference
|
|
|
|
APM_Compass.SetDeclination(ToRad(DECLINATION)); // set local difference between magnetic north and true north
|
|
|
|
int counter = 0;
|
|
|
|
|
2010-12-08 08:19:21 -04:00
|
|
|
SerFlu();
|
2010-12-08 07:59:14 -04:00
|
|
|
while(1) {
|
2010-10-19 12:22:58 -03:00
|
|
|
static float min[3], max[3], offset[3];
|
|
|
|
if((millis()- timer) > 100) {
|
|
|
|
timer = millis();
|
|
|
|
APM_Compass.Read();
|
|
|
|
APM_Compass.Calculate(0,0); // roll = 0, pitch = 0 for this example
|
|
|
|
|
|
|
|
// capture min
|
|
|
|
if( APM_Compass.Mag_X < min[0] ) min[0] = APM_Compass.Mag_X;
|
|
|
|
if( APM_Compass.Mag_Y < min[1] ) min[1] = APM_Compass.Mag_Y;
|
|
|
|
if( APM_Compass.Mag_Z < min[2] ) min[2] = APM_Compass.Mag_Z;
|
|
|
|
|
|
|
|
// capture max
|
|
|
|
if( APM_Compass.Mag_X > max[0] ) max[0] = APM_Compass.Mag_X;
|
|
|
|
if( APM_Compass.Mag_Y > max[1] ) max[1] = APM_Compass.Mag_Y;
|
|
|
|
if( APM_Compass.Mag_Z > max[2] ) max[2] = APM_Compass.Mag_Z;
|
|
|
|
|
|
|
|
// calculate offsets
|
|
|
|
offset[0] = -(max[0]+min[0])/2;
|
|
|
|
offset[1] = -(max[1]+min[1])/2;
|
|
|
|
offset[2] = -(max[2]+min[2])/2;
|
|
|
|
|
|
|
|
// display all to user
|
|
|
|
SerPri("Heading:");
|
|
|
|
SerPri(ToDeg(APM_Compass.Heading));
|
|
|
|
SerPri(" \t(");
|
|
|
|
SerPri(APM_Compass.Mag_X);
|
|
|
|
SerPri(",");
|
|
|
|
SerPri(APM_Compass.Mag_Y);
|
|
|
|
SerPri(",");
|
|
|
|
SerPri(APM_Compass.Mag_Z);
|
|
|
|
SerPri(")");
|
|
|
|
|
|
|
|
// display offsets
|
|
|
|
SerPri("\t offsets(");
|
|
|
|
SerPri(offset[0]);
|
|
|
|
SerPri(",");
|
|
|
|
SerPri(offset[1]);
|
|
|
|
SerPri(",");
|
|
|
|
SerPri(offset[2]);
|
|
|
|
SerPri(")");
|
|
|
|
SerPrln();
|
|
|
|
|
2010-12-08 07:59:14 -04:00
|
|
|
if(counter == 200) {
|
2010-10-19 12:22:58 -03:00
|
|
|
counter = 0;
|
|
|
|
SerPrln();
|
|
|
|
SerPrln("Roll and Rotate your quad untill offsets are not changing!");
|
2010-12-08 07:59:14 -04:00
|
|
|
// SerPrln("to exit from this loop, reboot your APM");
|
2010-10-19 12:22:58 -03:00
|
|
|
SerPrln();
|
|
|
|
delay(500);
|
|
|
|
}
|
|
|
|
counter++;
|
|
|
|
}
|
2010-12-08 07:59:14 -04:00
|
|
|
if(SerAva() > 0){
|
|
|
|
|
|
|
|
mag_offset_x = offset[0];
|
|
|
|
mag_offset_y = offset[1];
|
|
|
|
mag_offset_z = offset[2];
|
|
|
|
|
2010-12-08 08:19:21 -04:00
|
|
|
SerPriln("Saving Offsets to EEPROM");
|
2010-12-08 07:59:14 -04:00
|
|
|
writeEEPROM(mag_offset_x, mag_offset_x_ADR);
|
|
|
|
writeEEPROM(mag_offset_y, mag_offset_y_ADR);
|
|
|
|
writeEEPROM(mag_offset_z, mag_offset_z_ADR);
|
2010-12-08 08:19:21 -04:00
|
|
|
delay(500);
|
|
|
|
SerPriln("Saved...");
|
2010-12-08 07:59:14 -04:00
|
|
|
SerPrln();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2010-10-19 12:22:58 -03:00
|
|
|
}
|
|
|
|
#else
|
|
|
|
SerPrln("Magneto module is not activated on Arducopter.pde");
|
|
|
|
SerPrln("Check your program #definitions, upload firmware and try again!!");
|
|
|
|
// SerPrln("Now reboot your APM");
|
|
|
|
// for(;;)
|
|
|
|
// delay(10);
|
|
|
|
#endif
|
2010-10-18 15:24:46 -03:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-10-30 05:30:46 -03:00
|
|
|
/* ************************************************************** */
|
|
|
|
// Accell calibration
|
|
|
|
void CALIB_AccOffset() {
|
|
|
|
|
2010-12-08 07:59:14 -04:00
|
|
|
int loopy;
|
|
|
|
long xx = 0, xy = 0, xz = 0;
|
2010-10-30 05:30:46 -03:00
|
|
|
|
2010-12-08 07:59:14 -04:00
|
|
|
SerPrln("Initializing Accelerometers..");
|
2010-11-27 00:56:31 -04:00
|
|
|
adc.Init(); // APM ADC library initialization
|
2010-11-28 06:56:27 -04:00
|
|
|
// delay(250); // Giving small moment before starting
|
2010-10-30 05:30:46 -03:00
|
|
|
|
|
|
|
calibrateSensors(); // Calibrate neutral values of gyros (in Sensors.pde)
|
|
|
|
|
|
|
|
SerPrln();
|
2010-12-08 07:59:14 -04:00
|
|
|
SerPrln("Sampling 50 samples from Accelerometers, don't move your ArduCopter!");
|
2010-10-30 05:30:46 -03:00
|
|
|
SerPrln("Sample:\tAcc-X\tAxx-Y\tAcc-Z");
|
|
|
|
|
2010-12-08 07:59:14 -04:00
|
|
|
for(loopy = 1; loopy <= 50; loopy++) {
|
2010-10-30 05:30:46 -03:00
|
|
|
SerPri(" ");
|
|
|
|
SerPri(loopy);
|
|
|
|
SerPri(":");
|
|
|
|
tab();
|
2010-11-27 00:56:31 -04:00
|
|
|
SerPri(xx += adc.Ch(4));
|
2010-10-30 05:30:46 -03:00
|
|
|
tab();
|
2010-11-27 00:56:31 -04:00
|
|
|
SerPri(xy += adc.Ch(5));
|
2010-10-30 05:30:46 -03:00
|
|
|
tab();
|
2010-12-08 07:59:14 -04:00
|
|
|
SerPrln(xz += adc.Ch(6));
|
|
|
|
delay(20);
|
2010-10-30 05:30:46 -03:00
|
|
|
}
|
2010-11-28 06:56:27 -04:00
|
|
|
|
2010-10-30 05:30:46 -03:00
|
|
|
xx = xx / (loopy - 1);
|
|
|
|
xy = xy / (loopy - 1);
|
2010-11-28 13:20:16 -04:00
|
|
|
xz = xz / (loopy - 1);
|
2010-12-08 07:59:14 -04:00
|
|
|
xz = xz - 407; // Z-Axis correction
|
|
|
|
// xx += 42;
|
|
|
|
|
|
|
|
acc_offset_y = xy;
|
|
|
|
acc_offset_x = xx;
|
|
|
|
acc_offset_z = xz;
|
|
|
|
|
|
|
|
AN_OFFSET[3] = acc_offset_x;
|
|
|
|
AN_OFFSET[4] = acc_offset_y;
|
|
|
|
AN_OFFSET[5] = acc_offset_z;
|
|
|
|
|
|
|
|
SerPrln();
|
|
|
|
SerPrln("Offsets as follows: ");
|
2010-10-30 05:30:46 -03:00
|
|
|
SerPri(" ");
|
|
|
|
tab();
|
2010-12-08 07:59:14 -04:00
|
|
|
SerPri(acc_offset_x);
|
2010-10-30 05:30:46 -03:00
|
|
|
tab();
|
2010-12-08 07:59:14 -04:00
|
|
|
SerPri(acc_offset_y);
|
2010-10-30 05:30:46 -03:00
|
|
|
tab();
|
2010-12-08 07:59:14 -04:00
|
|
|
SerPrln(acc_offset_z);
|
|
|
|
|
|
|
|
SerPrln("Final results as follows: ");
|
|
|
|
SerPri(" ");
|
|
|
|
tab();
|
|
|
|
SerPri(adc.Ch(4) - acc_offset_x);
|
|
|
|
tab();
|
|
|
|
SerPri(adc.Ch(5) - acc_offset_y);
|
|
|
|
tab();
|
|
|
|
SerPrln(adc.Ch(6) - acc_offset_z);
|
|
|
|
SerPrln();
|
|
|
|
SerPrln("Final results should be close to 0, 0, 408 if they are far (-+10) from ");
|
2010-12-08 08:19:21 -04:00
|
|
|
SerPrln("those values, redo initialization or use Configurator for finetuning");
|
2010-12-08 07:59:14 -04:00
|
|
|
SerPrln();
|
|
|
|
SerPrln("Saving values to EEPROM");
|
|
|
|
writeEEPROM(acc_offset_x, acc_offset_x_ADR);
|
|
|
|
writeEEPROM(acc_offset_y, acc_offset_y_ADR);
|
|
|
|
writeEEPROM(acc_offset_z, acc_offset_z_ADR);
|
|
|
|
delay(200);
|
|
|
|
SerPrln("Saved..");
|
|
|
|
SerPrln();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void CALIB_Throttle() {
|
|
|
|
int tmpThrottle = 1100;
|
|
|
|
|
|
|
|
SerPrln("Move your throttle to MIN, reading starts in 3 seconds");
|
|
|
|
delay(1000);
|
|
|
|
SerPrln("2. ");
|
|
|
|
delay(1000);
|
|
|
|
SerPrln("1. ");
|
|
|
|
delay(1000);
|
|
|
|
SerPrln("Reading Throttle value, hit enter to exit");
|
|
|
|
|
|
|
|
SerFlu();
|
|
|
|
while(1) {
|
|
|
|
ch_throttle = APM_RC.InputCh(CH_THROTTLE);
|
|
|
|
SerPri("Throttle channel: ");
|
|
|
|
SerPrln(ch_throttle);
|
|
|
|
if(tmpThrottle > ch_throttle) tmpThrottle = ch_throttle;
|
|
|
|
delay(50);
|
|
|
|
if(SerAva() > 0){
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-10-30 05:30:46 -03:00
|
|
|
SerPriln();
|
2010-12-08 07:59:14 -04:00
|
|
|
SerPri("Lowest throttle value: ");
|
|
|
|
SerPrln(tmpThrottle);
|
|
|
|
SerPrln();
|
|
|
|
SerPrln("Saving MIN_THROTTLE to EEPROM");
|
|
|
|
writeEEPROM(tmpThrottle, MIN_THROTTLE_ADR);
|
|
|
|
delay(200);
|
|
|
|
SerPrln("Saved..");
|
|
|
|
SerPrln();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void CALIB_Esc() {
|
|
|
|
|
|
|
|
SerPrln("Disconnect your battery if you have it connected, keep your USB cable/Xbee connected!");
|
|
|
|
SerPrln("After battery is disconnected hit enter key and wait more instructions:");
|
|
|
|
SerPrln("As safety measure, unmount all your propellers before continuing!!");
|
|
|
|
|
|
|
|
WaitSerialEnter();
|
|
|
|
|
|
|
|
SerPrln("Move your Throttle to maximum and connect your battery. ");
|
|
|
|
SerPrln("after you hear beep beep tone, move your throttle to minimum and");
|
|
|
|
SerPrln("hit enter after you are ready to disarm motors.");
|
|
|
|
SerPrln("Arming now all motors");
|
|
|
|
delay(500);
|
|
|
|
SerPrln("Motors: ARMED");
|
|
|
|
delay(200);
|
|
|
|
SerPrln("Connect your battery and let ESCs to reboot!");
|
|
|
|
while(1) {
|
|
|
|
ch_throttle = APM_RC.InputCh(CH_THROTTLE);
|
|
|
|
APM_RC.OutputCh(0, ch_throttle);
|
|
|
|
APM_RC.OutputCh(1, ch_throttle);
|
|
|
|
APM_RC.OutputCh(2, ch_throttle);
|
|
|
|
APM_RC.OutputCh(3, ch_throttle);
|
|
|
|
|
|
|
|
// InstantPWM => Force inmediate output on PWM signals
|
|
|
|
APM_RC.Force_Out0_Out1();
|
|
|
|
APM_RC.Force_Out2_Out3();
|
|
|
|
delay(20);
|
|
|
|
if(SerAva() > 0){
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
APM_RC.OutputCh(0, 900);
|
|
|
|
APM_RC.OutputCh(1, 900);
|
|
|
|
APM_RC.OutputCh(2, 900);
|
|
|
|
APM_RC.OutputCh(3, 90);
|
|
|
|
APM_RC.Force_Out0_Out1();
|
|
|
|
APM_RC.Force_Out2_Out3();
|
2010-10-19 12:22:58 -03:00
|
|
|
|
2010-12-08 07:59:14 -04:00
|
|
|
SerPrln("Motors: DISARMED");
|
|
|
|
SerPrln();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Show_Settings() {
|
|
|
|
// Reading current EEPROM values
|
2010-12-08 08:19:21 -04:00
|
|
|
|
|
|
|
SerPrln("ArduCopter - Current settings");
|
|
|
|
SerPrln("-----------------------------");
|
|
|
|
SerPri("Firmware: ");
|
|
|
|
SerPri(VER);
|
|
|
|
SerPrln();
|
|
|
|
SerPrln();
|
|
|
|
|
2010-12-08 07:59:14 -04:00
|
|
|
readUserConfig();
|
|
|
|
delay(50);
|
|
|
|
|
|
|
|
SerPri("Magnetom. offsets (x,y,z): ");
|
|
|
|
SerPri(mag_offset_x);
|
|
|
|
cspc();
|
|
|
|
SerPri(mag_offset_y);
|
|
|
|
cspc();
|
|
|
|
SerPri(mag_offset_z);
|
|
|
|
SerPrln();
|
|
|
|
|
|
|
|
SerPri("Accel offsets (x,y,z): ");
|
|
|
|
SerPri(acc_offset_x);
|
|
|
|
cspc();
|
|
|
|
SerPri(acc_offset_y);
|
|
|
|
cspc();
|
|
|
|
SerPri(acc_offset_z);
|
|
|
|
SerPrln();
|
|
|
|
|
|
|
|
SerPri("Min Throttle: ");
|
|
|
|
SerPriln(MIN_THROTTLE);
|
|
|
|
|
2010-12-08 08:19:21 -04:00
|
|
|
SerPri("Magnetometer 1-ena/0-dis: ");
|
|
|
|
SerPriln(MAGNETOMETER, DEC);
|
|
|
|
|
|
|
|
SerPri("Camera mode: ");
|
|
|
|
SerPriln(cam_mode, DEC);
|
|
|
|
|
|
|
|
SerPrln();
|
|
|
|
SerPrln();
|
|
|
|
|
2010-12-08 07:59:14 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void cspc() {
|
|
|
|
SerPri(", ");
|
|
|
|
}
|
|
|
|
|
|
|
|
void WaitSerialEnter() {
|
|
|
|
// Flush serials
|
|
|
|
SerFlu();
|
|
|
|
delay(50);
|
|
|
|
while(1) {
|
|
|
|
if(SerAva() > 0){
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
delay(20);
|
|
|
|
}
|
|
|
|
delay(250);
|
|
|
|
SerFlu();
|
2010-10-30 05:30:46 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-11-28 06:56:27 -04:00
|
|
|
|
|
|
|
|