/**************************************************************************** * apps/system/adb/logcat_service.c * * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed with * this work for additional information regarding copyright ownership. The * ASF licenses this file to you 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. * ****************************************************************************/ /**************************************************************************** * Included Files ****************************************************************************/ #include #include #include #include #include "adb.h" #include "logcat_service.h" #include "hal/hal_uv_priv.h" /**************************************************************************** * Private types ****************************************************************************/ typedef struct logcat_service_s { adb_service_t service; uv_poll_t poll; int wait_ack; } logcat_service_t; /**************************************************************************** * Private Function Prototypes ****************************************************************************/ static void logcat_on_data_available(uv_poll_t * handle, int status, int events); /**************************************************************************** * Private Functions ****************************************************************************/ static int logcat_on_write(adb_service_t *service, apacket *p) { UNUSED(p); UNUSED(service); return -1; } static void logcat_on_kick(struct adb_service_s *service) { logcat_service_t *svc = container_of(service, logcat_service_t, service); if (!svc->wait_ack) { int ret; ret = uv_poll_start(&svc->poll, UV_READABLE, logcat_on_data_available); assert(ret == 0); } } static int logcat_on_ack(adb_service_t *service, apacket *p) { UNUSED(p); logcat_service_t *svc = container_of(service, logcat_service_t, service); svc->wait_ack = 0; logcat_on_kick(service); return 0; } static void close_cb(uv_handle_t *handle) { logcat_service_t *service = container_of(handle, logcat_service_t, poll); free(service); } static void logcat_on_close(struct adb_service_s *service) { int fd; int ret; logcat_service_t *svc = container_of(service, logcat_service_t, service); ret = uv_fileno((uv_handle_t *)&svc->poll, &fd); assert(ret == 0); close(fd); uv_close((uv_handle_t *)&svc->poll, close_cb); } static const adb_service_ops_t g_logcat_ops = { .on_write_frame = logcat_on_write, .on_ack_frame = logcat_on_ack, .on_kick = logcat_on_kick, .on_close = logcat_on_close }; static void logcat_on_data_available(uv_poll_t * handle, int status, int events) { int ret; int fd; apacket_uv_t *ap; logcat_service_t *service = container_of(handle, logcat_service_t, poll); adb_client_uv_t *client = (adb_client_uv_t *)handle->data; ap = adb_uv_packet_allocate(client, 0); if (ap == NULL) { uv_poll_stop(handle); return; } if (status) { adb_err("status error %d\n", status); /* Fatal error, stop service */ goto exit_stop_service; } ret = uv_fileno((uv_handle_t *)handle, &fd); assert(ret == 0); ret = read(fd, ap->p.data, CONFIG_ADBD_PAYLOAD_SIZE); if (ret < 0) { adb_err("frame read failed %d %d\n", ret, errno); if (errno == EAGAIN) { /* TODO this should never happen */ goto exit_release_packet; } /* Fatal error, stop service */ goto exit_stop_service; } if (ret == 0) { goto exit_release_packet; } service->wait_ack = 1; uv_poll_stop(handle); ap->p.write_len = ret; ap->p.msg.arg0 = service->service.id; ap->p.msg.arg1 = service->service.peer_id; adb_send_data_frame(&client->client, &ap->p); return; exit_release_packet: adb_hal_apacket_release(&client->client, &ap->p); return; exit_stop_service: adb_service_close(&client->client, &service->service, &ap->p); } /**************************************************************************** * Public Functions ****************************************************************************/ adb_service_t * logcat_service(adb_client_t *client, const char *params) { int ret; logcat_service_t *service = (logcat_service_t *)malloc(sizeof(logcat_service_t)); if (service == NULL) { return NULL; } service->service.ops = &g_logcat_ops; service->wait_ack = 0; /* TODO parse params string to extract logcat parameters */ ret = open(CONFIG_SYSLOG_DEVPATH, O_RDONLY | O_CLOEXEC); if (ret < 0) { adb_err("failed to open %s (%d)\n", CONFIG_SYSLOG_DEVPATH, errno); free(service); return NULL; } uv_handle_t *handle = adb_uv_get_client_handle(client); ret = uv_poll_init(handle->loop, &service->poll, ret); assert(ret == 0); service->poll.data = client; logcat_on_kick(&service->service); return &service->service; }