Всем привет, мы снова встретились, я ваш друг Цюаньчжаньцзюнь.
В этом разделе в основном описывается процесс подключения HelloDrone и процесс разрешения протокола Mavlink.
Основное содержание:
controlTower = new ControlTower(context); drone = new Drone(context);
@Override
/** * Переопределить метод onCreate и запустить его перед методом onStart. */
protected void onCreate(Bundle savedInstanceState) {
//Вызов onCreate родительского класса
super.onCreate(savedInstanceState);
//Установим информацию для отображения в главном окне
setContentView(R.layout.activity_main);
Log.i("lxw","+++++");
//Получаем имя файла приложения
final Context context = getApplicationContext();
//Создаем элемент управления
controlTower = new ControlTower(context);
//Создаем экземпляр дрона
drone = new Drone(context);
//Выбор режима
this.modeSelector = (Spinner) findViewById(R.id.modeSelect);
//Добавляем событие прослушивания выбора режима
this.modeSelector.setOnItemSelectedListener(new Spinner.OnItemSelectedListener()
{
@Override
public void onItemSelected(AdapterView<?> parent, View view, int position, long id)
{
onFlightModeSelected(view);
}
@Override
public void onNothingSelected(AdapterView<?> parent)
{
// Do nothing
}
});
final Button takePic = (Button) findViewById(R.id.take_photo_button);
takePic.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
takePhoto();
}
});
final Button toggleVideo = (Button) findViewById(R.id.toggle_video_recording);
toggleVideo.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
toggleVideoRecording();
}
});
videoView = (TextureView) findViewById(R.id.video_content);
videoView.setSurfaceTextureListener(new TextureView.SurfaceTextureListener() {
@Override
public void onSurfaceTextureAvailable(SurfaceTexture surface, int width, int height) {
alertUser("Video display is available.");
startVideoStream.setEnabled(true);
startVideoStreamUsingObserver.setEnabled(true);
}
@Override
public void onSurfaceTextureSizeChanged(SurfaceTexture surface, int width, int height) {
}
@Override
public boolean onSurfaceTextureDestroyed(SurfaceTexture surface) {
startVideoStream.setEnabled(false);
startVideoStreamUsingObserver.setEnabled(false);
return true;
}
@Override
public void onSurfaceTextureUpdated(SurfaceTexture surface) {
}
});
startVideoStream = (Button) findViewById(R.id.start_video_stream);
startVideoStream.setEnabled(false);
startVideoStream.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
alertUser("Starting video stream.");
startVideoStream(new Surface(videoView.getSurfaceTexture()));
}
});
stopVideoStream = (Button) findViewById(R.id.stop_video_stream);
stopVideoStream.setEnabled(false);
stopVideoStream.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
alertUser("Stopping video stream.");
stopVideoStream();
}
});
startVideoStreamUsingObserver = (Button) findViewById(R.id.start_video_stream_using_observer);
startVideoStreamUsingObserver.setEnabled(false);
startVideoStreamUsingObserver.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
alertUser("Starting video stream using observer for video stream packets.");
startVideoStreamForObserver();
}
});
stopVideoStreamUsingObserver = (Button) findViewById(R.id.stop_video_stream_using_observer);
stopVideoStreamUsingObserver.setEnabled(false);
stopVideoStreamUsingObserver.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
alertUser("Stopping video stream using observer for video stream packets.");
stopVideoStreamForObserver();
}
});
// Initialize media codec manager to decode video stream packets.
HandlerThread mediaCodecHandlerThread = new HandlerThread("MediaCodecHandlerThread");
mediaCodecHandlerThread.start();
Handler mediaCodecHandler = new Handler(mediaCodecHandlerThread.getLooper());
mediaCodecManager = new MediaCodecManager(mediaCodecHandler);
mainHandler = new Handler(getApplicationContext().getMainLooper());
}
@Override
public void onStart()
{
Log.i("lxw","Hello Drone на старте");
super.onStart();
//Выполняем задачи подключения
Log.i(MAGTAG,"onStart");
Log.i(MAGTAG,"this:"+this); //this:hello.MainActivity
Log.i(MAGTAG,"this2:"+getApplicationContext()); //this2:hello.StarterApplication
//Выполняем подключение контроллера
this.controlTower.connect(this,getApplicationContext());
//Обновляем тип режима
updateVehicleModesForType(this.droneType);
Log.i(MAGTAG,"connect");
}
Основной код:
this.controlTower.connect(this,getApplicationContext());
Реализация привязки службы o3drServicesConnection
public void connect(TowerListener listener,Context context) {
boolean value;
Log.i("LXW","AAA");
//Здесь нет мониторинга, условное суждение
if (towerListener != null && (isServiceConnecting.get() || isTowerConnected()))
return;
//пусто
if (listener == null)
{
throw new IllegalArgumentException("ServiceListener argument cannot be null.");
}
Log.i("LXW","CCC");
towerListener = listener;
Log.i("LXW","DDD");
if (!isTowerConnected() && !isServiceConnecting.get())
{
Intent serviceIntent = ApiAvailability.getInstance().getAvailableServicesInstance(context);
//value1:Intent { act=com.o3dr.services.android.lib.model.IDroidPlannerServices
// cmp=com.o3dr.sample.hellodrone/org.droidplanner.services.android.impl.api.DroidPlannerService }
Log.i("LXW","value1:"+ApiAvailability.getInstance().getAvailableServicesInstance(context));
value= this.context.bindService(serviceIntent, o3drServicesConnection, Context.BIND_AUTO_CREATE);
//value2:true
Log.i("LXW","value2:"+value);
isServiceConnecting.set(value);
}
Log.i("LXW","EEE");
}
После привязки службы первым делом нужно запустить onCreate(), а затем метод onBind().
Нам нужно объявить службу в файле манифеста.
Давайте посмотрим на код DroidPlannerService и обнаружим, что он наследует Service, который также соответствует требованиям сервиса.
/** Реализация фонового сервиса Android DroneKit. * DroneKit-Android background service implementation. */
public class DroidPlannerService extends Service {
/** * Status bar notification id */
private static final int FOREGROUND_ID = 101;
/** * Set of actions to notify the local app's components of the service events. */
public static final String ACTION_DRONE_CREATED = Utils.PACKAGE_NAME + ".ACTION_DRONE_CREATED";
public static final String ACTION_DRONE_DESTROYED = Utils.PACKAGE_NAME + ".ACTION_DRONE_DESTROYED";
public static final String ACTION_RELEASE_API_INSTANCE = Utils.PACKAGE_NAME + ".action.RELEASE_API_INSTANCE";
public static final String EXTRA_API_INSTANCE_APP_ID = "extra_api_instance_app_id";
/** * Used to broadcast service events. */
private LocalBroadcastManager lbm;
/** * Stores drone api instances per connected client. The client are denoted by their app id. */
final ConcurrentHashMap<String, DroneApi> droneApiStore = new ConcurrentHashMap<>();
/** * Caches drone managers per connection type. */
final ConcurrentHashMap<ConnectionParameter, DroneManager> droneManagers = new ConcurrentHashMap<>();
private DPServices dpServices;
private CameraInfoLoader cameraInfoLoader;
private List<CameraDetail> cachedCameraDetails;
/** * Generate a drone api instance for the client denoted by the given app id. * * @param listener Used to retrieve api information. * @param appId Application id of the connecting client. * @return a IDroneApi instance */
DroneApi registerDroneApi(IApiListener listener, String appId) {
if (listener == null)
return null;
DroneApi droneApi = new DroneApi(this, listener, appId);
droneApiStore.put(appId, droneApi);
lbm.sendBroadcast(new Intent(ACTION_DRONE_CREATED));
updateForegroundNotification();
return droneApi;
}
/** * Release the drone api instance attached to the given app id. * * @param appId Application id of the disconnecting client. */
void releaseDroneApi(String appId) {
if (appId == null)
return;
DroneApi droneApi = droneApiStore.remove(appId);
if (droneApi != null) {
Timber.d("Releasing drone api instance for " + appId);
droneApi.destroy();
lbm.sendBroadcast(new Intent(ACTION_DRONE_DESTROYED));
updateForegroundNotification();
}
}
/** * Установите соединение с автомобилем, используя заданные параметры соединения. * Establish a connection with a vehicle using the given connection parameter. * * @param connParams Параметры для подключения к автомобилю --- Параметры used to connect to the vehicle. * @param appId Идентификатор клиента подключающегося приложения --- Приложение id of the connecting client. * @param listener Обратный вызов для получения событий дрона. Перезвонить to receive drone events. * @return Экземпляр DroneManager, который действует как маршрутизатор между подключенными транспортными средствами и прослушивающими клиентами. * A DroneManager instance which acts as router between the connected vehicle and the listeneing client(s). */
DroneManager connectDroneManager(ConnectionParameter connParams, String appId, DroneApi listener) {
if (connParams == null || TextUtils.isEmpty(appId) || listener == null)
{
Log.i("lxw","connectDroneManager NULL");
return null;
}
DroneManager droneMgr = droneManagers.get(connParams);
Log.i("lxw","MYdroneMgr:"+droneMgr);
if (droneMgr == null) {
final DroneManager temp = DroneManager.generateDroneManager(getApplicationContext(), connParams, new Handler(Looper.getMainLooper()));
Log.i("lxw","temp:"+temp);
droneMgr = droneManagers.putIfAbsent(connParams, temp);
Log.i("lxw","MYdroneMgr2:"+droneMgr);
if(droneMgr == null){
Log.i("lxw","Generating:");
Timber.d("Generating new drone manager.");
droneMgr = temp;
}
else{
temp.destroy();
}
}
Timber.d("Drone manager connection for " + appId);
Log.i("lxw","Generating connect");
//устанавливаем соединение
droneMgr.connect(appId, listener, connParams);
return droneMgr;
}
/** * Disconnect the given client from the vehicle managed by the given drone manager. * * @param droneMgr Handler for the connected vehicle. * @param clientInfo Info of the disconnecting client. */
void disconnectDroneManager(DroneManager droneMgr, DroneApi.ClientInfo clientInfo) {
if (droneMgr == null || clientInfo == null || TextUtils.isEmpty(clientInfo.appId))
return;
String appId = clientInfo.appId;
Timber.d("Drone manager disconnection for " + appId);
droneMgr.disconnect(clientInfo);
if (droneMgr.getConnectedAppsCount() == 0) {
Timber.d("Destroying drone manager.");
droneMgr.destroy();
droneManagers.remove(droneMgr.getConnectionParameter());
}
}
/** * Retrieves the set of camera info provided by the app. * * @return a list of {@link CameraDetail} objects. */
synchronized List<CameraDetail> getCameraDetails() {
if (cachedCameraDetails == null) {
List<String> cameraInfoNames = cameraInfoLoader.getCameraInfoList();
List<CameraInfo> cameraInfos = new ArrayList<>(cameraInfoNames.size());
for (String infoName : cameraInfoNames) {
try {
cameraInfos.add(cameraInfoLoader.openFile(infoName));
} catch (Exception e) {
Timber.e(e, e.getMessage());
}
}
List<CameraDetail> cameraDetails = new ArrayList<>(cameraInfos.size());
for (CameraInfo camInfo : cameraInfos) {
cameraDetails.add(new CameraDetail(camInfo.name, camInfo.sensorWidth,
camInfo.sensorHeight, camInfo.sensorResolution, camInfo.focalLength,
camInfo.overlap, camInfo.sidelap, camInfo.isInLandscapeOrientation));
}
cachedCameraDetails = cameraDetails;
}
return cachedCameraDetails;
}
@Override
//Здесь возвращается ноль?
public IBinder onBind(Intent intent) {
Log.d("lxw","onBind service " + intent);
final String action = intent.getAction();
if (IDroidPlannerServices.class.getName().equals(action))
{
// Return binder to ipc client-server interaction.
return dpServices;
} else
{
return null;
}
}
@SuppressLint("NewApi")
@Override
public void onCreate() {
super.onCreate();
Log.d("lxw","Binding DroidPlannerService onCreate " );
Timber.d("Creating DroneKit-Android.");
final Context context = getApplicationContext();
dpServices = new DPServices(this);
lbm = LocalBroadcastManager.getInstance(context);
this.cameraInfoLoader = new CameraInfoLoader(context);
updateForegroundNotification();
}
@SuppressLint("NewApi")
private void updateForegroundNotification() {
final Context context = getApplicationContext();
//Put the service in the foreground
final NotificationCompat.Builder notifBuilder = new NotificationCompat.Builder(context)
.setContentTitle("DroneKit-Android")
.setPriority(NotificationCompat.PRIORITY_MIN)
.setSmallIcon(R.drawable.ic_stat_notify);
final int connectedCount = droneApiStore.size();
if (connectedCount > 1) {
notifBuilder.setContentText(connectedCount + " connected apps");
}
final Notification notification = notifBuilder.build();
startForeground(FOREGROUND_ID, notification);
}
@Override
public void onDestroy() {
super.onDestroy();
Timber.d("Destroying DroneKit-Android.");
for (DroneApi droneApi : droneApiStore.values()) {
droneApi.destroy();
}
droneApiStore.clear();
for (DroneManager droneMgr : droneManagers.values()) {
droneMgr.destroy();
}
droneManagers.clear();
dpServices.destroy();
stopForeground(true);
//Disable this service. It'll be reenabled the next time its local client needs it.
enableDroidPlannerService(getApplicationContext(), false);
}
@Override
public int onStartCommand(Intent intent, int flags, int startId) {
Log.i("lxw","MYintent:"+intent);
if (intent != null) {
final String action = intent.getAction();
switch (action) {
case ACTION_RELEASE_API_INSTANCE:
final String appId = intent.getStringExtra(EXTRA_API_INSTANCE_APP_ID);
releaseDroneApi(appId);
break;
}
}
stopSelf();
return START_NOT_STICKY;
}
/** * Toggles the DroidPlannerService component * @param context * @param enable */
public static void enableDroidPlannerService(Context context, boolean enable){
final ComponentName serviceComp = new ComponentName(context, DroidPlannerService.class);
final int newState = enable ? PackageManager.COMPONENT_ENABLED_STATE_ENABLED
: PackageManager.COMPONENT_ENABLED_STATE_DISABLED;
context.getPackageManager().setComponentEnabledSetting(serviceComp, newState, PackageManager.DONT_KILL_APP);
}
}
После выполнения onbind он вернется
Тогда оно будет выполнено
/** * Создать службу подключения */
private final ServiceConnection o3drServicesConnection = new ServiceConnection() {
@Override
public void onServiceConnected(ComponentName name, IBinder service) {
isServiceConnecting.set(false);
Log.i("lxw","onServiceConnected");
o3drServices = IDroidPlannerServices.Stub.asInterface(service);
try {
o3drServices.asBinder().linkToDeath(binderDeathRecipient, 0);
Log.i("lxw","onServiceConnected try ");
//Уведомление о соединении
notifyTowerConnected();
} catch (RemoteException e) {
//Сообщаем, что нет соединения
notifyTowerDisconnected();
}
}
@Override
public void onServiceDisconnected(ComponentName name) {
Log.i("lxw","EEE777");
isServiceConnecting.set(false);
notifyTowerDisconnected();
}
};
Основная функция: notifyTowerConnected()
/** * соединение для уведомлений */
void notifyTowerConnected()
{
Log.i("lxw","notifyTowerConnected");
if (towerListener == null)
return;
//Соединение вызова
Log.i("lxw","onTower");
towerListener.onTowerConnected();
}
Мы сосредоточимся на этой функции: TowerListener.onTowerConnected(); она в основном реализует регистрацию дронов и регистрацию слушателей.
public void onTowerConnected() {
//Запрос об успешном подключении
alertUser("DroneKit-Android Connected");
this.controlTower.registerDrone(this.drone, this.handler);
Log.i("lxw","onTowerConnected");
this.drone.registerDroneListener(this);
Log.i("lxw","registerDroneListener");
}
MainActivity реализует событие нажатия кнопки
public void onBtnConnectTap(View view)
{
//Определите, подключен ли статус соединения. Если он подключен, при нажатии будет предложено подключиться к порту.
if (this.drone.isConnected())
{
Log.i("lxw","Открыть соединение");
this.drone.disconnect();
} else
{
Log.i("lxw","Подключиться");
//Spinner предоставляет возможность быстрого выбора значения из коллекции данных. По умолчанию Spinner отображает текущее выбранное значение.
// При нажатии на счетчик появится всплывающее меню, содержащее все доступные значения, из которого можно выбрать новое значение для счетчика.
Spinner connectionSelector = (Spinner) findViewById(R.id.selectConnectionType);
//Получаем свойства раскрывающегося списка
int selectedConnectionType = connectionSelector.getSelectedItemPosition();
//Определяем, является ли текущее соединение USB-соединением, если да, то устанавливаем USB-соединение, иначе это UDP-соединение
//conntParams:ConnectionParameter{connectionType=0, paramsBundle=[extra_usb_baud_rate=115200]}
ConnectionParameter connectionParams = selectedConnectionType == ConnectionType.TYPE_USB
? ConnectionParameter.newUsbConnection(null)
: ConnectionParameter.newUdpConnection(null);
Log.i("lxw","connectionParams:"+connectionParams);
//Внутренний класс реализует события подключения
this.drone.connect(connectionParams, new LinkListener()
{
@Override
public void onLinkStateUpdated(@NonNull LinkConnectionStatus connectionStatus)
{
Log.i("lxw","connectionStatus:"+connectionStatus);
}
});
}
}
Основной код: анонимный внутренний класс реализует события подключения.
//Анонимный внутренний класс реализует события подключения
this.drone.connect(connectionParams, new LinkListener()
{
@Override
public void onLinkStateUpdated(@NonNull LinkConnectionStatus connectionStatus)
{
Log.i("lxw","connectionStatus:"+connectionStatus);
}
});
public void connect(ConnectionParameter connParams, LinkListener linkListener) {
Log.i("lxw","Начать операцию подключения дрона");
//USB-соединение
VehicleApi.getApi(this).connect(connParams);
this.connectionParameter = connParams;
this.linkListener = linkListener;
Log.i("lxw","this.connectionParameter:"+this.connectionParameter);
Log.i("lxw","this.linkListener:"+this.linkListener);
}
Продолжайте изучать основной код: VehicleApi.getApi(this).connect(connParams);
public void connect(ConnectionParameter parameter) {
Bundle params = new Bundle();
//params:Bundle[{}]
Log.i("lxw","Bundle params:"+params);
params.putParcelable(EXTRA_CONNECT_PARAMETER, parameter);
Action connectAction = new Action(ACTION_CONNECT, params);
//дрон выполняет асинхронные операции
//com.o3dr.services.android.lib.model.action.Action
Log.i("lxw","Bundle connectAction:"+connectAction);
drone.performAsyncAction(connectAction);
}
Формат:drone.performAsyncAction(connectAction);
public boolean performAsyncAction(Action action) {
//Поток асинхронного выполнения
return performAsyncActionOnDroneThread(action, null);
}
Код анализа: PerformAsyncActionOnDroneThread.
public boolean performAsyncActionOnDroneThread(Action action, AbstractCommandListener listener) {
//Выполняем асинхронную обработку поведения
return performAsyncActionOnHandler(action, this.handler, listener);
}
Процедура выполненияAsyncActionOnHandler
public boolean performAsyncActionOnHandler(Action action, Handler handler, AbstractCommandListener listener) {
boolean value=false;
Log.i("lxw","performAsyncActionOnHandler");
final IDroneApi droneApi = droneApiRef.get();
//org.droidplanner.services.android.impl.api.DroneApi
Log.i("lxw","droneApi:"+droneApi);
value=isStarted(droneApi);
Log.i("lxw","isStarted value:"+value);
if (value)
{
try {
//Выполнение поведения
Log.i("lxw","executeAsyncAction:");
droneApi.executeAsyncAction(action, wrapListener(handler, listener));
return true;
} catch (RemoteException e)
{
Log.i("lxw","Выполнить RemoteException");
Toast.makeText(getContext(), «Асинхронное выполнение 2», Toast.LENGTH_LONG).show();
handleRemoteException(e);
}
}
Log.i("lxw","disconnect7");
return false;
}
Основной код, вызывающий беспокойство:droneApi.executeAsyncAction(action, WrapListener(handler, Listener));
/** * поведение выполнения */
public void executeAction(Action action, ICommandListener listener) throws RemoteException {
//com.o3dr.services.android.lib.model.action.Action
Log.i("lxw","executeAction action:"+action);
if (action == null)
{
Log.i("lxw","action null:");
return;
}
String type = action.getType();
//type com.o3dr.services.android.action.CONNECT
Log.i("lxw","type:"+type);
if (type == null)
{
Log.i("lxw","type null:");
return;
}
Bundle data = action.getData();
Log.i("lxw","Bundle data:"+data);
if (data != null) {
Log.i("lxw","data not null:");
data.setClassLoader(context.getClassLoader());
}
//Получаем устройство
Drone drone = getDrone();
Log.i("lxw","drone1:"+drone);
Log.i("lxw","mytype:"+type);
switch (type) {
// CONNECTION ACTIONS
case ConnectionActions.ACTION_CONNECT:
ConnectionParameter parameter = data.getParcelable(ConnectionActions.EXTRA_CONNECT_PARAMETER);
Log.i("lxw","ConnectionActions:");
//устанавливаем соединение
connect(parameter);
break;
case ConnectionActions.ACTION_DISCONNECT:
disconnect();
break;
// CAMERA ACTIONS
case CameraActions.ACTION_START_VIDEO_STREAM: {
Surface videoSurface = data.getParcelable(CameraActions.EXTRA_VIDEO_DISPLAY);
String videoTag = data.getString(CameraActions.EXTRA_VIDEO_TAG, "");
Bundle videoProps = data.getBundle(CameraActions.EXTRA_VIDEO_PROPERTIES);
if (videoProps == null) {
//Only case where it's null is when interacting with a deprecated client version.
//In this case, we assume that the client is attempting to start a solo stream, since that's
//the only api that was exposed.
videoProps = new Bundle();
videoProps.putInt(CameraActions.EXTRA_VIDEO_PROPS_UDP_PORT, VideoManager.ARTOO_UDP_PORT);
}
CommonApiUtils.startVideoStream(drone, videoProps, ownerId, videoTag, videoSurface, listener);
break;
}
case ExperimentalActions.ACTION_START_VIDEO_STREAM_FOR_OBSERVER: {
String videoTag = data.getString(CameraActions.EXTRA_VIDEO_TAG, "");
CommonApiUtils.startVideoStreamForObserver(drone, ownerId, videoTag, listener);
break;
}
case CameraActions.ACTION_STOP_VIDEO_STREAM: {
String videoTag = data.getString(CameraActions.EXTRA_VIDEO_TAG, "");
CommonApiUtils.stopVideoStream(drone, ownerId, videoTag, listener);
break;
}
case ExperimentalActions.ACTION_STOP_VIDEO_STREAM_FOR_OBSERVER: {
String videoTag = data.getString(CameraActions.EXTRA_VIDEO_TAG, "");
CommonApiUtils.stopVideoStreamForObserver(drone, ownerId, videoTag, listener);
break;
}
// MISSION ACTIONS
case MissionActions.ACTION_BUILD_COMPLEX_MISSION_ITEM:
if (drone instanceof MavLinkDrone || drone == null) {
CommonApiUtils.buildComplexMissionItem((MavLinkDrone) drone, data);
} else {
CommonApiUtils.postErrorEvent(CommandExecutionError.COMMAND_UNSUPPORTED, listener);
}
break;
case MissionActions.ACTION_SAVE_MISSION: {
Mission mission = data.getParcelable(MissionActions.EXTRA_MISSION);
Uri saveUri = data.getParcelable(MissionActions.EXTRA_SAVE_MISSION_URI);
if (saveUri == null) {
CommonApiUtils.postErrorEvent(CommandExecutionError.COMMAND_FAILED, listener);
} else {
MissionUtils.saveMission(context, mission, saveUri, listener);
}
break;
}
case MissionActions.ACTION_LOAD_MISSION: {
Uri loadUri = data.getParcelable(MissionActions.EXTRA_LOAD_MISSION_URI);
boolean setMission = data.getBoolean(MissionActions.EXTRA_SET_LOADED_MISSION, false);
if (loadUri != null) {
Mission mission = MissionUtils.loadMission(context, loadUri);
if(mission != null){
// Going back to the caller.
data.putParcelable(MissionActions.EXTRA_MISSION, mission);
if(setMission){
Bundle params = new Bundle();
params.putParcelable(EXTRA_MISSION, mission);
params.putBoolean(EXTRA_PUSH_TO_DRONE, false);
executeAction(new Action(ACTION_SET_MISSION, params), listener);
}
}
}
break;
}
default:
if (droneMgr != null) {
droneMgr.executeAsyncAction(clientInfo, action, listener);
} else {
CommonApiUtils.postErrorEvent(CommandExecutionError.COMMAND_FAILED, listener);
}
break;
}
}
Здесь выбираем USB-соединение и ориентируемся на выполняемые функции.
/** * Выполнение задач по подключению * @param connParams */
public void connect(ConnectionParameter connParams) {
try {
Log.i("lxw","connParams:"+connParams);
//Validate the given connection parameter
connParams = checkConnectionParameter(connParams);
Log.i("lxw","connParams1:"+connParams);
//Validate the current connection parameter for the drone
ConnectionParameter currentConnParams = this.connectionParams == null
? this.connectionParams
: checkConnectionParameter(this.connectionParams);
Log.i("lxw","currentConnParams:"+currentConnParams);
if (!connParams.equals(currentConnParams))
{
Log.i("lxw","droneMgr:"+droneMgr);
if (this.droneMgr != null)
{
Log.i("lxw","droneMgr is not null:");
LinkConnectionStatus connectionStatus = LinkConnectionStatus
.newFailedConnectionStatus(LinkConnectionStatus.ADDRESS_IN_USE,
"Connection already started with different connection parameters");
onConnectionStatus(connectionStatus);
return;
}
this.connectionParams = connParams;
Log.i("lxw","this.connectionParams:"+this.connectionParams);
//ownerId:com.o3dr.sample.hellodrone
Log.i("lxw","ownerId:"+ownerId);
//устанавливаем вызов управления соединением
this.droneMgr = service.connectDroneManager(this.connectionParams, ownerId, this);
Log.i("lxw","this.droneMgr:"+this.droneMgr);
Log.i("lxw","bufdata:"+isEventsBufferingEnabled());
if(isEventsBufferingEnabled())
{
Log.i("lxw","isEventsBufferingEnabled:");
eventsBuffer.clear();
handler.postDelayed(eventsDispatcher, this.connectionParams.getEventsDispatchingPeriod());
}
}
} catch (ConnectionException e)
{
Log.i("lxw","link connect:");
LinkConnectionStatus connectionStatus = LinkConnectionStatus
.newFailedConnectionStatus(LinkConnectionStatus.INVALID_CREDENTIALS, e.getMessage());
onConnectionStatus(connectionStatus);
disconnect();
}
}
Пример: this.droneMgr = service.connectDroneManager(this.connectionParams, OwnerId, this);
/** * Установите соединение с автомобилем, используя заданные параметры соединения. * Establish a connection with a vehicle using the given connection parameter. * * @param connParams Параметры для подключения к автомобилю --- Параметры used to connect to the vehicle. * @param appId Идентификатор клиента подключающегося приложения --- Приложение id of the connecting client. * @param listener Обратный вызов для получения событий дрона. Перезвонить to receive drone events. * @return Экземпляр DroneManager, который действует как маршрутизатор между подключенными транспортными средствами и прослушивающими клиентами. * A DroneManager instance which acts as router between the connected vehicle and the listeneing client(s). */
DroneManager connectDroneManager(ConnectionParameter connParams, String appId, DroneApi listener) {
if (connParams == null || TextUtils.isEmpty(appId) || listener == null)
{
Log.i("lxw","connectDroneManager NULL");
return null;
}
DroneManager droneMgr = droneManagers.get(connParams);
Log.i("lxw","MYdroneMgr:"+droneMgr);
if (droneMgr == null) {
final DroneManager temp = DroneManager.generateDroneManager(getApplicationContext(), connParams, new Handler(Looper.getMainLooper()));
Log.i("lxw","temp:"+temp);
droneMgr = droneManagers.putIfAbsent(connParams, temp);
Log.i("lxw","MYdroneMgr2:"+droneMgr);
if(droneMgr == null){
Log.i("lxw","Generating:");
Timber.d("Generating new drone manager.");
droneMgr = temp;
}
else{
temp.destroy();
}
}
Timber.d("Drone manager connection for " + appId);
Log.i("lxw","Generating connect");
//устанавливаем соединение
droneMgr.connect(appId, listener, connParams);
return droneMgr;
}
Код анализа:droneMgr.connect(appId, прослушиватель, connParams);
public synchronized void connect(String appId, DroneApi listener, ConnectionParameter connParams) {
//org.droidplanner.services.android.impl.api.DroneApi
Log.i("lxw","listener:"+listener);
Log.i("lxw","listener2:"+TextUtils.isEmpty(appId));
if (listener == null || TextUtils.isEmpty(appId)) {
Log.i("lxw","listener return:");
return;
}
connectedApps.put(appId, listener);
//Начало подключения
Log.i("lxw","doConnect:");
doConnect(appId, listener, connParams);
}
Пример: doConnect(appId, прослушиватель, connParams);
protected void doConnect(String appId, DroneApi listener, ConnectionParameter connParams) {
Log.i("lxw"," MavlinkDrone Manager doConnect:");
if (mavClient.isDisconnected()) {
Timber.i("Opening connection for %s", appId);
mavClient.openConnection();
Log.i("lxw"," mavClientdoConnect:1");
} else {
if (isConnected()) {
listener.onDroneEvent(DroneInterfaces.DroneEventsType.CONNECTED, drone);
if (!drone.isConnectionAlive())
listener.onDroneEvent(DroneInterfaces.DroneEventsType.HEARTBEAT_TIMEOUT, drone);
}
Log.i("lxw"," mavClientdoConnect:2");
}
mavClient.registerForTLogLogging(appId, connParams.getTLogLoggingUri());
Log.i("lxw"," updateDroneStreamRate:");
updateDroneStreamRate(connParams);
}
Сосредоточьтесь на: mavClient.openConnection();
public synchronized void openConnection()
{
if(isConnected() || isConnecting())
{
Log.i("lxw"," openConnection:1");
return;
}
final String tag = toString();
//Создаем Mavlink----Создать the mavlink connection
final int connectionType = connParams.getConnectionType();
final Bundle paramsBundle = connParams.getParamsBundle();
Log.i("lxw"," connectionType:"+connectionType); //0
Log.i("lxw"," paramsBundle:"+paramsBundle); //скорость передачи данных
Log.i("lxw"," mavlinkConn:"+mavlinkConn); //UsbConnection
if(mavlinkConn == null) {
switch (connectionType) {
//Тип USB
case ConnectionType.TYPE_USB: //0
final int baudRate = paramsBundle.getInt(ConnectionType.EXTRA_USB_BAUD_RATE,
ConnectionType.DEFAULT_USB_BAUD_RATE);
mavlinkConn = new UsbConnection(context, baudRate);
Log.i("lxw"," mavlinkConn:6"+mavlinkConn);
Timber.i("Connecting over usb.");
break;
//Тип Bluetooth
case ConnectionType.TYPE_BLUETOOTH:
//Retrieve the bluetooth address to connect to
final String bluetoothAddress = paramsBundle.getString(ConnectionType.EXTRA_BLUETOOTH_ADDRESS);
mavlinkConn = new BluetoothConnection(context, bluetoothAddress);
Timber.i("Connecting over bluetooth.");
break;
//Тип TCP
case ConnectionType.TYPE_TCP:
//Retrieve the server ip and port
final String tcpServerIp = paramsBundle.getString(ConnectionType.EXTRA_TCP_SERVER_IP);
final int tcpServerPort = paramsBundle.getInt(ConnectionType
.EXTRA_TCP_SERVER_PORT, ConnectionType.DEFAULT_TCP_SERVER_PORT);
mavlinkConn = new AndroidTcpConnection(context, tcpServerIp, tcpServerPort, new WifiConnectionHandler(context));
Timber.i("Connecting over tcp.");
break;
//тип UDP
case ConnectionType.TYPE_UDP:
final int udpServerPort = paramsBundle
.getInt(ConnectionType.EXTRA_UDP_SERVER_PORT, ConnectionType.DEFAULT_UDP_SERVER_PORT);
mavlinkConn = new AndroidUdpConnection(context, udpServerPort, new WifiConnectionHandler(context));
Timber.i("Connecting over udp.");
break;
//Тип СОЛО
case ConnectionType.TYPE_SOLO: {
Timber.i("Creating solo connection");
final String soloLinkId = paramsBundle.getString(ConnectionType.EXTRA_SOLO_LINK_ID, null);
final String linkPassword = paramsBundle.getString(ConnectionType.EXTRA_SOLO_LINK_PASSWORD, null);
mavlinkConn = new SoloConnection(context, soloLinkId, linkPassword);
break;
}
default:
Timber.e("Unrecognized connection type: %s", connectionType);
return;
}
}
Log.i("lxw"," mavlinkConn:7");
Log.i("lxw"," mConnectionListener:"+mConnectionListener);
mavlinkConn.addMavLinkConnectionListener(tag, mConnectionListener);
//Check if we need to ping a server to receive UDP data stream.
if (connectionType == ConnectionType.TYPE_UDP)
{
final String pingIpAddress = paramsBundle.getString(ConnectionType.EXTRA_UDP_PING_RECEIVER_IP);
if (!TextUtils.isEmpty(pingIpAddress)) {
try {
final InetAddress resolvedAddress = InetAddress.getByName(pingIpAddress);
final int pingPort = paramsBundle.getInt(ConnectionType.EXTRA_UDP_PING_RECEIVER_PORT);
final long pingPeriod = paramsBundle.getLong(ConnectionType.EXTRA_UDP_PING_PERIOD,
ConnectionType.DEFAULT_UDP_PING_PERIOD);
final byte[] pingPayload = paramsBundle.getByteArray(ConnectionType.EXTRA_UDP_PING_PAYLOAD);
((AndroidUdpConnection) mavlinkConn).addPingTarget(resolvedAddress, pingPort, pingPeriod, pingPayload);
} catch (UnknownHostException e) {
Timber.e(e, "Unable to resolve UDP ping server ip address.");
}
}
}
Log.i("lxw"," mavlinkConn:888");
if (mavlinkConn.getConnectionStatus() == MavLinkConnection.MAVLINK_DISCONNECTED)
{
Log.i("lxw"," Начать выполнение Mavlink соединять");
mavlinkConn.connect(null);
}
}
Ключевые функции: выполнить связанный с mavlink mavlinkConn.connect(null);
public void connect(Bundle extras) {
Log.i("lxw"," mavlinkConn:999");
if (mConnectionStatus.compareAndSet(MAVLINK_DISCONNECTED, MAVLINK_CONNECTING)) {
extrasHolder.set(extras);
mLogger.logInfo(TAG, "Starting connection thread.");
Log.i("lxw"," создание задачи подключения mavlink");
mConnectThread = new Thread(mConnectingTask, "MavLinkConnection-Connecting Thread");
//Начало соединения
mConnectThread.start();
//Сообщаем о состоянии ссылки
reportConnecting();
}
}
/** * Запустите процесс подключения. Начинать the connection process. */
private final Runnable mConnectingTask = new Runnable() {
@Override
public void run() {
// Load the connection specific preferences
loadPreferences();
// Open the connection
try {
Log.i("lxw"," Запустите выполнение, чтобы открыть openConnection. run");
openConnection(extrasHolder.get());
Log.i("lxw"," mavlinkConn:111");
} catch (IOException e) {
Log.i("lxw"," mavlinkConn:2222");
// Ignore errors while shutting down
if (mConnectionStatus.get() != MAVLINK_DISCONNECTED) {
reportIOException(e);
mLogger.logErr(TAG, e);
}
disconnect();
}
mLogger.logInfo(TAG, "Exiting connecting thread.");
}
};
Продолжайте рассматривать код: openConnection(extrasHolder.get());
@Override
protected void openConnection(Bundle connectionExtras) throws IOException {
boolean id_device;
Log.i("lxw"," mUsbConnection:"+mUsbConnection);
if (mUsbConnection != null) {
try {
Log.i("lxw"," BBB123");
mUsbConnection.openUsbConnection(connectionExtras);
Log.d(TAG, "Reusing previous usb connection.");
return;
} catch (IOException e) {
Log.i("lxw"," BBB456");
Log.e(TAG, "Previous usb connection is not usable.", e);
mUsbConnection = null;
}
}
id_device=isFTDIdevice(context);
Log.i("lxw","id_device:"+id_device);
if (id_device)
{
final UsbConnectionImpl tmp = new UsbFTDIConnection(context, this, mBaudRate);
try {
tmp.openUsbConnection(connectionExtras);
// If the call above is successful, 'mUsbConnection' will be set.
mUsbConnection = tmp;
Log.d(TAG, "Using FTDI usb connection.");
} catch (IOException e) {
Log.d(TAG, "Unable to open a ftdi usb connection. Falling back to the open "
+ "usb-library.", e);
}
}
// Fallback
if (mUsbConnection == null) {
Log.i("lxw"," KKK");
Log.i("lxw"," BT:"+mBaudRate);
Log.i("lxw"," BT2:"+context);
final UsbConnectionImpl tmp = new UsbCDCConnection(context, this, mBaudRate);
Log.i("lxw"," BT3:"+tmp);
// If an error happens here, let it propagate up the call chain since this is the fallback.
tmp.openUsbConnection(connectionExtras);
Log.i("lxw"," BT5:");
mUsbConnection = tmp;
Log.i("lxw"," BT6:"+mUsbConnection);
Log.d(TAG, "Using open-source usb connection.");
}
}
Продолжайте рассматривать код: tmp.openUsbConnection(connectionExtras);
protected void openUsbConnection(Bundle extras) throws IOException {
Log.i("lxw"," cdc");
extrasHolder.set(extras);
registerUsbPermissionBroadcastReceiver();
// Get UsbManager from Android.
UsbManager manager = (UsbManager) mContext.getSystemService(Context.USB_SERVICE);
//Get the list of available devices
List<UsbDevice> availableDevices = UsbSerialProber.getAvailableSupportedDevices(manager);
Log.i("lxw"," availableDevices:"+availableDevices);
if (availableDevices.isEmpty())
{
Log.i("lxw"," CDC NULL");
Log.d(TAG, "No Devices found");
throw new IOException("No Devices found");
}
//Pick the first device
UsbDevice device = availableDevices.get(0);
if (manager.hasPermission(device))
{
openUsbDevice(device, extras);
} else {
removeWatchdog();
scheduler = Executors.newSingleThreadScheduledExecutor();
scheduler.schedule(permissionWatchdog, 15, TimeUnit.SECONDS);
Log.d(TAG, "Requesting permission to access usb device " + device.getDeviceName());
manager.requestPermission(device, usbPermissionIntent);
}
}
Продолжайте просматривать код: openUsbDevice(device, extras);
private void openUsbDevice(UsbDevice device, Bundle extras) throws IOException {
//Получаем UsbManager с Android. --- Get UsbManager from Android.
UsbManager manager = (UsbManager) mContext.getSystemService(Context.USB_SERVICE);
//Находим первый доступный драйвер. --- Find the first available driver.
final UsbSerialDriver serialDriver = UsbSerialProber.openUsbDevice(manager, device);
if (serialDriver == null) {
Log.d(TAG, "No Devices found");
throw new IOException("No Devices found");
} else {
Log.d(TAG, "Opening using Baud rate " + mBaudRate);
try {
serialDriver.open();
serialDriver.setParameters(mBaudRate, 8, UsbSerialDriver.STOPBITS_1, UsbSerialDriver.PARITY_NONE);
serialDriverRef.set(serialDriver);
onUsbConnectionOpened(extras);
} catch (IOException e)
{
Log.e(TAG, "Error setting up device: " + e.getMessage(), e);
try {
serialDriver.close();
} catch (IOException e2) {
// Ignore.
}
}
}
}
продолжатьonUsbConnectionOpened (дополнительно);
protected void onUsbConnectionOpened(Bundle extras) {
parentConnection.onConnectionOpened(extras);
}
/** * Откройте устройство и выполните задачу mManagerTask. * @param extras */
protected void onConnectionOpened(Bundle extras) {
if (mConnectionStatus.compareAndSet(MAVLINK_CONNECTING, MAVLINK_CONNECTED)) {
extrasHolder.set(extras);
mLogger.logInfo(TAG, "Starting manager thread.");
Log.i("lxw"," mManagerTask в основном обрабатывает информацию mavlink");
mTaskThread = new Thread(mManagerTask, "MavLinkConnection-Manager Thread");
mTaskThread.start();
}
}
Анонимный внутренний класс реализует окончательную обработку протокола mavlink, связанную с
private final Runnable mManagerTask = new Runnable() {
@Override
public void run() {
Thread sendingThread = null;
Thread loggingThread = null;
try {
final long connectionTime = System.currentTimeMillis();
mConnectionTime.set(connectionTime);
reportConnect(connectionTime);
// Launch the 'Sending' thread
mLogger.logInfo(TAG, "Starting sender thread.");
Log.i("lxw"," Starting new thread.");
sendingThread = new Thread(mSendingTask, "MavLinkConnection-Sending Thread");
sendingThread.start();
//Launch the 'Logging' thread
mLogger.logInfo(TAG, "Starting logging thread.");
Log.i("lxw"," Starting sender thread.");
loggingThread = new Thread(mLoggingTask, "MavLinkConnection-Logging Thread");
loggingThread.start();
//Анализ протокола
final Parser parser = new Parser();
//com.MAVLink.Parser
Log.i("lxw"," parser:"+parser);
//Статус парсинга сервиса
parser.stats.resetStats();
//Создаем массив разобранных данных
final byte[] readBuffer = new byte[READ_BUFFER_SIZE];
while (mConnectionStatus.get() == MAVLINK_CONNECTED)
{
//Читаем размер буфера
int bufferSize = readDataBlock(readBuffer);
//bufferSize=
Log.i("lxw"," bufferSize:"+bufferSize);
handleData(parser, bufferSize, readBuffer);
}
} catch (IOException e) {
// Ignore errors while shutting down
if (mConnectionStatus.get() != MAVLINK_DISCONNECTED) {
reportIOException(e);
mLogger.logErr(TAG, e);
}
} finally {
if (sendingThread != null && sendingThread.isAlive()) {
sendingThread.interrupt();
}
if (loggingThread != null && loggingThread.isAlive()) {
loggingThread.interrupt();
}
disconnect();
mLogger.logInfo(TAG, "Exiting manager thread.");
}
}
/** * Начать обработку данных * @param parser * @param bufferSize * @param buffer */
private void handleData(Parser parser, int bufferSize, byte[] buffer) {
if (bufferSize < 1) {
return;
}
for (int i = 0; i < bufferSize; i++)
{
//анализируем пакет
MAVLinkPacket receivedPacket = parser.mavlink_parse_char(buffer[i] & 0x00ff);
if (receivedPacket != null)
{
queueToLog(receivedPacket);
Log.i("lxw"," Начать разбор отчета: ");
reportReceivedPacket(receivedPacket);
}
}
}
};
Среди них наиболее важный контент, отправляемый приложением во внешний мир: Среди них наиболее важный контент, отправляемый приложением во внешний мир: Среди них наиболее важный контент, отправляемый приложением во внешний мир:
private final Runnable mSendingTask = new Runnable() {
@Override
public void run() {
try {
while (mConnectionStatus.get() == MAVLINK_CONNECTED) {
byte[] buffer = mPacketsToSend.take();
try {
sendBuffer(buffer);
queueToLog(buffer);
} catch (IOException e) {
reportIOException(e);
mLogger.logErr(TAG, e);
}
}
} catch (InterruptedException e) {
mLogger.logVerbose(TAG, e.getMessage());
} finally {
disconnect();
}
}
};
Наше внимание здесь по-прежнему сосредоточено на приеме.
Основная обработка: handleData(parser, bufferSize, readBuffer); Основная обработка: handleData(parser, bufferSize, readBuffer); Основная обработка: handleData(parser, bufferSize, readBuffer);
private void handleData(Parser parser, int bufferSize, byte[] buffer) {
if (bufferSize < 1) {
return;
}
for (int i = 0; i < bufferSize; i++)
{
//анализируем пакет
MAVLinkPacket receivedPacket = parser.mavlink_parse_char(buffer[i] & 0x00ff);
if (receivedPacket != null)
{
queueToLog(receivedPacket);
Log.i("lxw"," Начать разбор отчета: ");
reportReceivedPacket(receivedPacket);
}
}
}
private void reportReceivedPacket(MAVLinkPacket packet)
{
//Не является ли слушатель ненулевым
if (mListeners.isEmpty())
{
return;
}
for (MavLinkConnectionListener listener : mListeners.values())
{
Log.i("lxw"," onReceivePacket:");
//Получаем обработку данных
listener.onReceivePacket(packet);
}
}
public void onReceivePacket(final MAVLinkPacket packet) {
Log.i("lxw"," notifyReceivedData:");
listener.notifyReceivedData(packet);
Log.i("lxw"," receivedMsg END:");
}
public void notifyReceivedData(MAVLinkPacket packet) {
MAVLinkMessage receivedMsg = packet.unpack();
Log.i("lxw"," receivedMsg:"+receivedMsg);
if (receivedMsg == null)
return;
Log.i("lxw"," MSGID:"+receivedMsg.msgid);
if (receivedMsg.msgid == msg_command_ack.MAVLINK_MSG_ID_COMMAND_ACK)
{
Log.i("lxw"," command ack:");
msg_command_ack commandAck = (msg_command_ack) receivedMsg;
handleCommandAck(commandAck);
} else
{
Log.i("lxw"," Handler receiveData :");
this.mavLinkMsgHandler.receiveData(receivedMsg);
if (this.drone != null)
{
Log.i("lxw"," Ядро обрабатывает информацию mavlink: ");
this.drone.onMavLinkMessageReceived(receivedMsg);
}
}
if (!connectedApps.isEmpty())
{
Log.i("lxw"," spp :"+connectedApps.values());
//org.droidplanner.services.android.impl.api.DroneApi
for (DroneApi droneEventsListener : connectedApps.values())
{
droneEventsListener.onReceivedMavLinkMessage(receivedMsg);
}
}
}
Здесь есть две важные части кода: протокол с ответом и протокол без ответа. В основном мы фокусируемся на протоколе без ответа.
Сначала вызовите метод в Ardupilot:
public void onMavLinkMessageReceived(MAVLinkMessage message) {
Log.i("lxw","ardupilot message.sysid :"+message.sysid );
if ((message.sysid != this.getSysid()) && !isMavLinkMessageException(message)) {
// Reject Messages that are not for the system id
return;
}
// Filter Components IDs to be specifically the IDs that can be processed
int compId = message.compid;
if (compId != AUTOPILOT_COMPONENT_ID
&& compId != ARTOO_COMPONENT_ID
&& compId != SiK_RADIO_FIXED_COMPID ){
return;
}
if (!getParameterManager().processMessage(message)) {
getWaypointManager().processMessage(message);
getCalibrationSetup().processMessage(message);
switch (message.msgid) {
case msg_statustext.MAVLINK_MSG_ID_STATUSTEXT:
// These are any warnings sent from APM:Copter with
// gcs_send_text_P()
// This includes important thing like arm fails, prearm fails, low
// battery, etc.
// also less important things like "erasing logs" and
// "calibrating barometer"
msg_statustext msg_statustext = (msg_statustext) message;
processStatusText(msg_statustext);
break;
case msg_vfr_hud.MAVLINK_MSG_ID_VFR_HUD:
processVfrHud((msg_vfr_hud) message);
break;
case msg_raw_imu.MAVLINK_MSG_ID_RAW_IMU:
msg_raw_imu msg_imu = (msg_raw_imu) message;
mag.newData(msg_imu);
break;
case msg_radio.MAVLINK_MSG_ID_RADIO:
msg_radio m_radio = (msg_radio) message;
processSignalUpdate(m_radio.rxerrors, m_radio.fixed, m_radio.rssi,
m_radio.remrssi, m_radio.txbuf, m_radio.noise, m_radio.remnoise);
break;
case msg_rc_channels_raw.MAVLINK_MSG_ID_RC_CHANNELS_RAW:
rc.setRcInputValues((msg_rc_channels_raw) message);
break;
case msg_servo_output_raw.MAVLINK_MSG_ID_SERVO_OUTPUT_RAW:
rc.setRcOutputValues((msg_servo_output_raw) message);
break;
case msg_camera_feedback.MAVLINK_MSG_ID_CAMERA_FEEDBACK:
getCamera().newImageLocation((msg_camera_feedback) message);
break;
case msg_mount_status.MAVLINK_MSG_ID_MOUNT_STATUS:
processMountStatus((msg_mount_status) message);
break;
case msg_named_value_int.MAVLINK_MSG_ID_NAMED_VALUE_INT:
processNamedValueInt((msg_named_value_int) message);
break;
//*************** Magnetometer calibration messages handling *************//
case msg_mag_cal_progress.MAVLINK_MSG_ID_MAG_CAL_PROGRESS:
case msg_mag_cal_report.MAVLINK_MSG_ID_MAG_CAL_REPORT:
getMagnetometerCalibration().processCalibrationMessage(message);
break;
default:
break;
}
}
Log.i("lxw","super message.sysid :"+message.sysid );
//Вот метод вызова родительского класса
super.onMavLinkMessageReceived(message);
}
public void onMavLinkMessageReceived(MAVLinkMessage message) {
Log.i("lxw","message.sysid :"+message.sysid );
if ( (message.sysid != this.getSysid()) && !isMavLinkMessageException(message) )
{
// Reject messages that are not for this drone's system id
return;
}
onHeartbeat(message);
switch (message.msgid) {
case msg_radio_status.MAVLINK_MSG_ID_RADIO_STATUS:
msg_radio_status m_radio_status = (msg_radio_status) message;
processSignalUpdate(m_radio_status.rxerrors, m_radio_status.fixed, m_radio_status.rssi,
m_radio_status.remrssi, m_radio_status.txbuf, m_radio_status.noise, m_radio_status.remnoise);
break;
case msg_attitude.MAVLINK_MSG_ID_ATTITUDE:
msg_attitude m_att = (msg_attitude) message;
processAttitude(m_att);
break;
case msg_heartbeat.MAVLINK_MSG_ID_HEARTBEAT:
msg_heartbeat msg_heart = (msg_heartbeat) message;
processHeartbeat(msg_heart);
break;
case msg_vibration.MAVLINK_MSG_ID_VIBRATION:
msg_vibration vibrationMsg = (msg_vibration) message;
processVibrationMessage(vibrationMsg);
break;
//*************** EKF State handling ******************//
case msg_ekf_status_report.MAVLINK_MSG_ID_EKF_STATUS_REPORT:
processEfkStatus((msg_ekf_status_report) message);
break;
case msg_sys_status.MAVLINK_MSG_ID_SYS_STATUS:
msg_sys_status m_sys = (msg_sys_status) message;
processSysStatus(m_sys);
break;
case msg_global_position_int.MAVLINK_MSG_ID_GLOBAL_POSITION_INT:
processGlobalPositionInt((msg_global_position_int) message);
break;
case msg_gps_raw_int.MAVLINK_MSG_ID_GPS_RAW_INT:
processGpsState((msg_gps_raw_int) message);
break;
case msg_mission_item.MAVLINK_MSG_ID_MISSION_ITEM:
processHomeUpdate((msg_mission_item) message);
break;
case msg_mission_current.MAVLINK_MSG_ID_MISSION_CURRENT:
missionStats.setWpno(((msg_mission_current) message).seq);
break;
case msg_mission_item_reached.MAVLINK_MSG_ID_MISSION_ITEM_REACHED:
missionStats.setLastReachedWaypointNumber(((msg_mission_item_reached) message).seq);
break;
case msg_nav_controller_output.MAVLINK_MSG_ID_NAV_CONTROLLER_OUTPUT:
msg_nav_controller_output m_nav = (msg_nav_controller_output) message;
setDisttowpAndSpeedAltErrors(m_nav.wp_dist, m_nav.alt_error, m_nav.aspd_error);
break;
}
}
Это общий процесс обработки протокола, который будет подробно проанализирован позже.
Издатель: Full stack программист и руководитель стека, укажите источник для перепечатки: https://javaforall.cn/170568.html Исходная ссылка: https://javaforall.cn