| /* |
| * Copyright (c) 2021, The Linux Foundation. All rights reserved. |
| * |
| * Redistribution and use in source and binary forms, with or without |
| * modification, are permitted provided that the following conditions are |
| * met: |
| * * Redistributions of source code must retain the above copyright |
| * notice, this list of conditions and the following disclaimer. |
| * * Redistributions in binary form must reproduce the above |
| * copyright notice, this list of conditions and the following |
| * disclaimer in the documentation and/or other materials provided |
| * with the distribution. |
| * * Neither the name of The Linux Foundation nor the names of its |
| * contributors may be used to endorse or promote products derived |
| * from this software without specific prior written permission. |
| * |
| * THIS SOFTWARE IS PROVIDED "AS IS" AND ANY EXPRESS OR IMPLIED |
| * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF |
| * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT |
| * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS |
| * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR |
| * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF |
| * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR |
| * BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, |
| * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE |
| * OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN |
| * IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
| */ |
| |
| #define LOG_TAG "PAL: StreamUltraSound" |
| |
| #include "StreamUltraSound.h" |
| #include "Session.h" |
| #include "SessionAlsaPcm.h" |
| #include "ResourceManager.h" |
| #include "Device.h" |
| #include "us_detect_api.h" |
| #include <unistd.h> |
| |
| StreamUltraSound::StreamUltraSound(const struct pal_stream_attributes *sattr __unused, struct pal_device *dattr __unused, |
| const uint32_t no_of_devices __unused, const struct modifier_kv *modifiers __unused, |
| const uint32_t no_of_modifiers __unused, const std::shared_ptr<ResourceManager> rm): |
| StreamCommon(sattr,dattr,no_of_devices,modifiers,no_of_modifiers,rm) |
| { |
| session->registerCallBack((session_callback)HandleCallBack,((uint64_t) this)); |
| rm->registerStream(this); |
| } |
| |
| StreamUltraSound::~StreamUltraSound() |
| { |
| rm->resetStreamInstanceID(this); |
| rm->deregisterStream(this); |
| } |
| |
| int32_t StreamUltraSound::stop() |
| { |
| int32_t status = 0; |
| |
| mStreamMutex.lock(); |
| PAL_DBG(LOG_TAG, "Enter. session handle - %pK mStreamAttr->direction - %d state %d", |
| session, mStreamAttr->direction, currentState); |
| |
| if (currentState == STREAM_STARTED || currentState == STREAM_PAUSED) { |
| |
| status = session->setParameters(this, DEVICE_POP_SUPPRESSOR, |
| PAL_PARAM_ID_ULTRASOUND_RAMPDOWN, NULL); |
| if (0 != status) { |
| PAL_ERR(LOG_TAG, "SetParameters failed for Rampdown, status = %d", status); |
| } |
| /* Adjust the delay based on requirement */ |
| usleep(20000); |
| |
| for (int i = 0; i < mDevices.size(); i++) { |
| rm->deregisterDevice(mDevices[i], this); |
| } |
| PAL_VERBOSE(LOG_TAG, "In %s, device count - %zu", |
| GET_DIR_STR(mStreamAttr->direction), mDevices.size()); |
| |
| rm->lockGraph(); |
| status = session->stop(this); |
| if (0 != status) { |
| PAL_ERR(LOG_TAG, "Error:%s session stop failed with status %d", |
| GET_DIR_STR(mStreamAttr->direction), status); |
| } |
| PAL_VERBOSE(LOG_TAG, "session stop successful"); |
| for (int32_t i=0; i < mDevices.size(); i++) { |
| status = mDevices[i]->stop(); |
| if (0 != status) { |
| PAL_ERR(LOG_TAG, "Error:%s device stop failed with status %d", |
| GET_DIR_STR(mStreamAttr->direction), status); |
| } |
| } |
| rm->unlockGraph(); |
| PAL_VERBOSE(LOG_TAG, "devices stop successful"); |
| currentState = STREAM_STOPPED; |
| } else if (currentState == STREAM_STOPPED || currentState == STREAM_IDLE) { |
| PAL_INFO(LOG_TAG, "Stream is already in Stopped state %d", currentState); |
| } else { |
| PAL_ERR(LOG_TAG, "Error:Stream should be in start/pause state, %d", currentState); |
| status = -EINVAL; |
| } |
| PAL_DBG(LOG_TAG, "Exit. status %d, state %d", status, currentState); |
| |
| mStreamMutex.unlock(); |
| return status; |
| } |
| |
| |
| |
| int32_t StreamUltraSound::setParameters(uint32_t param_id, void *payload) |
| { |
| int32_t status = 0; |
| |
| if (!payload) |
| { |
| status = -EINVAL; |
| PAL_ERR(LOG_TAG, "invalid params"); |
| goto error; |
| } |
| |
| mStreamMutex.lock(); |
| if (currentState == STREAM_IDLE) { |
| PAL_ERR(LOG_TAG, "Invalid stream state: IDLE for param ID: %d", param_id); |
| mStreamMutex.unlock(); |
| return -EINVAL; |
| } |
| // Stream may not know about tags, so use setParameters instead of setConfig |
| switch (param_id) { |
| case PAL_PARAM_ID_UPD_REGISTER_FOR_EVENTS: |
| { |
| status = session->setParameters(NULL, 0, param_id, payload); |
| if (status) |
| PAL_ERR(LOG_TAG, "Error:%d, Failed to setParam for registering an event", |
| status); |
| break; |
| } |
| default: |
| PAL_ERR(LOG_TAG, "Error:Unsupported param id %u", param_id); |
| status = -EINVAL; |
| break; |
| } |
| |
| mStreamMutex.unlock(); |
| PAL_DBG(LOG_TAG, "exit, session parameter %u set with status %d", param_id, status); |
| error: |
| return status; |
| } |
| |
| void StreamUltraSound::HandleEvent(uint32_t event_id, void *data, uint32_t event_size) { |
| struct event_id_upd_detection_event_t *event_info = nullptr; |
| event_info = (struct event_id_upd_detection_event_t *)data; |
| uint32_t event_type = event_info->proximity_event_type; |
| |
| PAL_INFO(LOG_TAG, "%s event received %d", |
| (event_type == US_DETECT_NEAR)? "NEAR": "FAR", event_type); |
| |
| if (callback_) { |
| PAL_INFO(LOG_TAG, "Notify detection event to client"); |
| mStreamMutex.lock(); |
| callback_((pal_stream_handle_t *)this, event_id, &event_type, |
| event_size, cookie_); |
| mStreamMutex.unlock(); |
| } |
| } |
| |
| void StreamUltraSound::HandleCallBack(uint64_t hdl, uint32_t event_id, |
| void *data, uint32_t event_size, uint32_t miid __unused) { |
| StreamUltraSound *StreamUPD = nullptr; |
| PAL_DBG(LOG_TAG, "Enter, event detected on SPF, event id = 0x%x, event size =%d", |
| event_id, event_size); |
| if (event_id == EVENT_ID_GENERIC_US_DETECTION) { |
| StreamUPD = (StreamUltraSound *)hdl; |
| StreamUPD->HandleEvent(event_id, data, event_size); |
| } |
| PAL_DBG(LOG_TAG, "Exit"); |
| } |