pebble/src/fw/process_management/launcher_app_message.c
2025-01-27 11:38:16 -08:00

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);
}