mirror of
https://github.com/google/pebble.git
synced 2025-05-22 19:34:51 +00:00
Import of the watch repository from Pebble
This commit is contained in:
commit
3b92768480
10334 changed files with 2564465 additions and 0 deletions
164
src/fw/process_management/launcher_app_message.c
Normal file
164
src/fw/process_management/launcher_app_message.c
Normal file
|
@ -0,0 +1,164 @@
|
|||
/*
|
||||
* 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);
|
||||
}
|
Loading…
Add table
Add a link
Reference in a new issue