mirror of
https://github.com/google/pebble.git
synced 2025-05-05 01:11:40 -04:00
164 lines
5.3 KiB
C
164 lines
5.3 KiB
C
/*
|
|
* Copyright 2024 Google LLC
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
#include "launcher_app_message.h"
|
|
|
|
#include "app_run_state.h"
|
|
#include "applib/app_message/app_message_internal.h"
|
|
#include "process_management/app_install_manager.h"
|
|
#include "system/logging.h"
|
|
#include "system/passert.h"
|
|
|
|
#include "util/dict.h"
|
|
|
|
#define LAUNCHER_MESSAGE_ENDPOINT_ID (0x31)
|
|
|
|
typedef enum {
|
|
//! Used as reply from the watch to the phone, to indicate the app is not running.
|
|
//! Or, when pushed from phone to watch, this value will have the effect of killing the app.
|
|
NOT_RUNNING_DEPRECATED = 0x00,
|
|
//! Used as reply from the watch to the phone, to indicate the app is running.
|
|
//! Or, when pushed from phone to watch, this value will have the effect of launching the app.
|
|
RUNNING_DEPRECATED = 0x01,
|
|
} AppStateDeprecated;
|
|
|
|
enum {
|
|
//! This key/value can be pushed from the phone to the watch to launch
|
|
//! or kill an app on the watch.
|
|
RUN_STATE_KEY = 0x01, // TUPLE_UINT8
|
|
STATE_FETCH_KEY = 0x02, // TUPLE_UINT8
|
|
};
|
|
|
|
static uint8_t s_transaction_id;
|
|
|
|
// For unit testing
|
|
void launcher_app_message_reset(void) {
|
|
s_transaction_id = 0;
|
|
}
|
|
|
|
void launcher_app_message_send_app_state_deprecated(const Uuid *uuid, bool running) {
|
|
// Deprecated: 0x31 endpoint, only used by Android versions < 2.2 and iOS
|
|
AppStateDeprecated app_state = running ? RUNNING_DEPRECATED : NOT_RUNNING_DEPRECATED;
|
|
|
|
uint8_t buffer[sizeof(AppMessagePush) + sizeof(Tuple) + sizeof(uint32_t)];
|
|
AppMessagePush *push_message = (AppMessagePush *)buffer;
|
|
|
|
*push_message = (const AppMessagePush) {
|
|
.header = {
|
|
.command = CMD_PUSH,
|
|
.transaction_id = s_transaction_id++,
|
|
},
|
|
.uuid = *uuid,
|
|
};
|
|
|
|
uint32_t size = sizeof(Dictionary) + sizeof(Tuple) + sizeof(uint32_t);
|
|
const Tuplet tuplet = TupletInteger(RUN_STATE_KEY, (uint32_t) app_state);
|
|
PBL_ASSERTN(DICT_OK == dict_serialize_tuplets_to_buffer(
|
|
&tuplet, 1, (uint8_t *)&push_message->dictionary, &size));
|
|
|
|
comm_session_send_data(comm_session_get_system_session(), LAUNCHER_MESSAGE_ENDPOINT_ID,
|
|
(const uint8_t *)buffer, sizeof(buffer), COMM_SESSION_DEFAULT_TIMEOUT);
|
|
}
|
|
|
|
static bool prv_has_invalid_length(size_t expected, size_t actual) {
|
|
if (actual < expected) {
|
|
PBL_LOG(LOG_LEVEL_ERROR, "Too short");
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
static bool prv_receive_push_cmd(CommSession *session,
|
|
AppMessagePush *push_message, size_t length) {
|
|
if (prv_has_invalid_length(sizeof(AppMessagePush), length)) {
|
|
return false;
|
|
}
|
|
|
|
bool success = false;
|
|
|
|
// Scan the dictionary:
|
|
const size_t dict_size = length - sizeof(AppMessagePush) + sizeof(Dictionary);
|
|
DictionaryIterator iter;
|
|
const Tuple *tuple = dict_read_begin_from_buffer(&iter,
|
|
(const uint8_t *) &push_message->dictionary,
|
|
dict_size);
|
|
while (tuple) {
|
|
uint8_t cmd = tuple->key;
|
|
switch (cmd) {
|
|
case RUN_STATE_KEY:
|
|
if (tuple->value->uint8 != RUNNING_DEPRECATED) {
|
|
cmd = APP_RUN_STATE_STOP_COMMAND;
|
|
} else {
|
|
cmd = APP_RUN_STATE_RUN_COMMAND;
|
|
}
|
|
success = true;
|
|
break;
|
|
case STATE_FETCH_KEY:
|
|
cmd = APP_RUN_STATE_STATUS_COMMAND;
|
|
success = true;
|
|
break;
|
|
default:
|
|
cmd = APP_RUN_STATE_INVALID_COMMAND;
|
|
}
|
|
|
|
// Call into app_run_state to take the action (to avoid duping code):
|
|
const Uuid *app_uuid = &push_message->uuid;
|
|
app_run_state_command(NULL, (AppRunStateCommand)cmd, app_uuid);
|
|
tuple = dict_read_next(&iter);
|
|
}
|
|
|
|
return success;
|
|
}
|
|
|
|
static void prv_send_ack_nack_reply(CommSession *session, const uint8_t transaction_id, bool ack) {
|
|
const AppMessageAck nack_message = {
|
|
.header = {
|
|
.command = ack ? CMD_ACK : CMD_NACK,
|
|
.transaction_id = transaction_id,
|
|
},
|
|
};
|
|
comm_session_send_data(session, LAUNCHER_MESSAGE_ENDPOINT_ID,
|
|
(const uint8_t *) &nack_message, sizeof(nack_message),
|
|
COMM_SESSION_DEFAULT_TIMEOUT);
|
|
}
|
|
|
|
void launcher_app_message_protocol_msg_callback_deprecated(CommSession *session,
|
|
const uint8_t* data, size_t length) {
|
|
if (prv_has_invalid_length(sizeof(AppMessageHeader), length)) {
|
|
return;
|
|
}
|
|
|
|
AppMessageHeader *message = (AppMessageHeader *) data;
|
|
bool ack = false;
|
|
switch (message->command) {
|
|
case CMD_PUSH: {
|
|
// Incoming message:
|
|
ack = prv_receive_push_cmd(session, (AppMessagePush *) message, length);
|
|
break;
|
|
}
|
|
|
|
case CMD_ACK:
|
|
case CMD_NACK:
|
|
// Ignore ACK / NACKs
|
|
return;
|
|
|
|
default:
|
|
// Ignore everything else
|
|
break;
|
|
}
|
|
|
|
prv_send_ack_nack_reply(session, message->transaction_id, ack);
|
|
}
|