/* Copyright (c) 2014-2016, The Linux Foundation. All rights reserved. * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License version 2 and * only version 2 as published by the Free Software Foundation. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. */ #include #include #include #include #include #include #include #include #include #include #include #include #include "diagchar.h" #include "diagfwd.h" #include "diag_mux.h" #include "diag_usb.h" #include "diag_memorydevice.h" struct diag_mux_state_t *diag_mux; static struct diag_logger_t usb_logger; static struct diag_logger_t md_logger; static struct diag_logger_ops usb_log_ops = { .open = diag_usb_connect_all, .close = diag_usb_disconnect_all, .queue_read = diag_usb_queue_read, .write = diag_usb_write, .close_peripheral = NULL }; static struct diag_logger_ops md_log_ops = { .open = diag_md_open_all, .close = diag_md_close_all, .queue_read = NULL, .write = diag_md_write, .close_peripheral = diag_md_close_peripheral, }; int diag_mux_init() { diag_mux = kzalloc(sizeof(struct diag_mux_state_t), GFP_KERNEL); if (!diag_mux) return -ENOMEM; kmemleak_not_leak(diag_mux); usb_logger.mode = DIAG_USB_MODE; usb_logger.log_ops = &usb_log_ops; md_logger.mode = DIAG_MEMORY_DEVICE_MODE; md_logger.log_ops = &md_log_ops; diag_md_init(); /* * Set USB logging as the default logger. This is the mode * Diag should be in when it initializes. */ diag_mux->usb_ptr = &usb_logger; diag_mux->md_ptr = &md_logger; diag_mux->logger = &usb_logger; diag_mux->mux_mask = 0; diag_mux->mode = DIAG_USB_MODE; return 0; } void diag_mux_exit() { kfree(diag_mux); } int diag_mux_register(int proc, int ctx, struct diag_mux_ops *ops) { int err = 0; if (!ops) return -EINVAL; if (proc < 0 || proc >= NUM_MUX_PROC) return 0; /* Register with USB logger */ usb_logger.ops[proc] = ops; err = diag_usb_register(proc, ctx, ops); if (err) { pr_err("diag: MUX: unable to register usb operations for proc: %d, err: %d\n", proc, err); return err; } md_logger.ops[proc] = ops; err = diag_md_register(proc, ctx, ops); if (err) { pr_err("diag: MUX: unable to register md operations for proc: %d, err: %d\n", proc, err); return err; } return 0; } int diag_mux_queue_read(int proc) { struct diag_logger_t *logger = NULL; if (proc < 0 || proc >= NUM_MUX_PROC) return -EINVAL; if (!diag_mux) return -EIO; if (diag_mux->mode == DIAG_MULTI_MODE) logger = diag_mux->usb_ptr; else logger = diag_mux->logger; if (logger && logger->log_ops && logger->log_ops->queue_read) return logger->log_ops->queue_read(proc); return 0; } int diag_mux_write(int proc, unsigned char *buf, int len, int ctx) { struct diag_logger_t *logger = NULL; int peripheral; if (proc < 0 || proc >= NUM_MUX_PROC) return -EINVAL; if (!diag_mux) return -EIO; peripheral = GET_BUF_PERIPHERAL(ctx); if (peripheral > NUM_PERIPHERALS) return -EINVAL; if (MD_PERIPHERAL_MASK(peripheral) & diag_mux->mux_mask) logger = diag_mux->md_ptr; else logger = diag_mux->usb_ptr; if (logger && logger->log_ops && logger->log_ops->write) return logger->log_ops->write(proc, buf, len, ctx); return 0; } int diag_mux_close_peripheral(int proc, uint8_t peripheral) { struct diag_logger_t *logger = NULL; if (proc < 0 || proc >= NUM_MUX_PROC) return -EINVAL; /* Peripheral should account for Apps data as well */ if (peripheral > NUM_PERIPHERALS) return -EINVAL; if (!diag_mux) return -EIO; if (MD_PERIPHERAL_MASK(peripheral) & diag_mux->mux_mask) logger = diag_mux->md_ptr; else logger = diag_mux->logger; if (logger && logger->log_ops && logger->log_ops->close_peripheral) return logger->log_ops->close_peripheral(proc, peripheral); return 0; } int diag_mux_switch_logging(int *req_mode, int *peripheral_mask) { unsigned int new_mask = 0; if (!req_mode) return -EINVAL; if (*peripheral_mask <= 0 || *peripheral_mask > DIAG_CON_ALL) { pr_err("diag: mask %d in %s\n", *peripheral_mask, __func__); return -EINVAL; } switch (*req_mode) { case DIAG_USB_MODE: new_mask = ~(*peripheral_mask) & diag_mux->mux_mask; if (new_mask != DIAG_CON_NONE) *req_mode = DIAG_MULTI_MODE; break; case DIAG_MEMORY_DEVICE_MODE: new_mask = (*peripheral_mask) | diag_mux->mux_mask; if (new_mask != DIAG_CON_ALL) *req_mode = DIAG_MULTI_MODE; break; default: pr_err("diag: Invalid mode %d in %s\n", *req_mode, __func__); return -EINVAL; } switch (diag_mux->mode) { case DIAG_USB_MODE: if (*req_mode == DIAG_MEMORY_DEVICE_MODE) { diag_mux->usb_ptr->log_ops->close(); diag_mux->logger = diag_mux->md_ptr; diag_mux->md_ptr->log_ops->open(); } else if (*req_mode == DIAG_MULTI_MODE) { diag_mux->md_ptr->log_ops->open(); diag_mux->logger = NULL; } break; case DIAG_MEMORY_DEVICE_MODE: if (*req_mode == DIAG_USB_MODE) { diag_mux->md_ptr->log_ops->close(); diag_mux->logger = diag_mux->usb_ptr; diag_mux->usb_ptr->log_ops->open(); } else if (*req_mode == DIAG_MULTI_MODE) { diag_mux->usb_ptr->log_ops->open(); diag_mux->logger = NULL; } break; case DIAG_MULTI_MODE: if (*req_mode == DIAG_USB_MODE) { diag_mux->md_ptr->log_ops->close(); diag_mux->logger = diag_mux->usb_ptr; } else if (*req_mode == DIAG_MEMORY_DEVICE_MODE) { diag_mux->usb_ptr->log_ops->close(); diag_mux->logger = diag_mux->md_ptr; } break; } diag_mux->mode = *req_mode; diag_mux->mux_mask = new_mask; *peripheral_mask = new_mask; return 0; }