aidl: Import QTI Vibrator AIDL
* CAF tag LA.UM.9.1.r1-11600.03-SMxxx0.QSSI12.0 Change-Id: I48b8326aab719b236ff6b39c05dc4df1eeb6f688
This commit is contained in:
committed by
Bruno Martins
parent
d0e36f6d22
commit
8e38bbd395
305
aidl/vibrator/VibratorOffload.cpp
Normal file
305
aidl/vibrator/VibratorOffload.cpp
Normal file
@@ -0,0 +1,305 @@
|
||||
/*
|
||||
* 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 "vendor.qti.vibrator.offload"
|
||||
|
||||
#include <stdio.h>
|
||||
#include <stdint.h>
|
||||
#include <string.h>
|
||||
#include <thread>
|
||||
#include <linux/input.h>
|
||||
#include <linux/slatecom_interface.h>
|
||||
#include <log/log.h>
|
||||
#include <fcntl.h>
|
||||
#include <cutils/log.h>
|
||||
#include <cutils/uevent.h>
|
||||
#include <sys/poll.h>
|
||||
#include <sys/ioctl.h>
|
||||
|
||||
#include "include/Vibrator.h"
|
||||
#include "VibratorPatterns.h"
|
||||
|
||||
extern "C" {
|
||||
#include "libsoc_helper.h"
|
||||
}
|
||||
|
||||
namespace aidl {
|
||||
namespace android {
|
||||
namespace hardware {
|
||||
namespace vibrator {
|
||||
|
||||
#define UEVENT_MSG_LEN 1024
|
||||
#define SLATE_EVENT "SLATE_EVENT="
|
||||
#define SLATE_EVENT_STRING_LEN 12 //length of SLATE_EVENT
|
||||
|
||||
PatternOffload::PatternOffload()
|
||||
{
|
||||
std::thread t(&PatternOffload::SSREventListener, this);
|
||||
t.detach();
|
||||
}
|
||||
|
||||
void PatternOffload::SSREventListener(void)
|
||||
{
|
||||
int device_fd, n, ssr_event = 0;
|
||||
char msg[UEVENT_MSG_LEN + 2];
|
||||
char *msg_ptr = msg;
|
||||
|
||||
/* Offload during the bootup */
|
||||
SendPatterns();
|
||||
|
||||
device_fd = uevent_open_socket(64*1024, true);
|
||||
if(device_fd < 0)
|
||||
{
|
||||
ALOGE("open socket failed: %d", device_fd);
|
||||
return;
|
||||
}
|
||||
|
||||
while ((n = uevent_kernel_multicast_recv(device_fd, msg, UEVENT_MSG_LEN)) > 0) {
|
||||
if (n <= 0 || n >= UEVENT_MSG_LEN) {
|
||||
ALOGE("Message length %d is not correct\n", n);
|
||||
continue;
|
||||
}
|
||||
msg[n] = '\0';
|
||||
msg[n+1] = '\0';
|
||||
if (strstr(msg, "slate_com_dev")) {
|
||||
while(*msg_ptr) {
|
||||
if(!strncmp(msg_ptr, SLATE_EVENT, SLATE_EVENT_STRING_LEN)) {
|
||||
msg_ptr += SLATE_EVENT_STRING_LEN;
|
||||
ssr_event = (atoi(msg_ptr));
|
||||
switch(ssr_event) {
|
||||
case SLATE_AFTER_POWER_UP:
|
||||
ALOGD("SLATE is powered up");
|
||||
SendPatterns();
|
||||
break;
|
||||
}
|
||||
}
|
||||
while(*msg_ptr++);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/** Offload patterns
|
||||
* The sequence of steps in offloading patterns.
|
||||
* 1. Open the Glink channel to offload the patterns
|
||||
* 2. Send the configuration/meta data to co-proc
|
||||
* 3. Wait for the response from the co-proc
|
||||
* 4. Send the pattern data to co-proc
|
||||
* 5. Wait for the response
|
||||
* 6. Exit
|
||||
*/
|
||||
void PatternOffload::SendPatterns()
|
||||
{
|
||||
uint8_t *data;
|
||||
uint32_t len;
|
||||
int32_t rc;
|
||||
|
||||
rc = initChannel();
|
||||
if (rc < 0)
|
||||
return;
|
||||
|
||||
rc = get_pattern_config(&data, &len);
|
||||
if (rc < 0 || !data)
|
||||
return;
|
||||
|
||||
/* Send config data */
|
||||
rc = sendData(data, len);
|
||||
if (rc < 0)
|
||||
return;
|
||||
|
||||
rc = get_pattern_data(&data, &len);
|
||||
if (rc < 0 || !data)
|
||||
return;
|
||||
|
||||
/* Send pattern data */
|
||||
rc = sendData(data, len);
|
||||
if (rc < 0)
|
||||
return;
|
||||
|
||||
free_pattern_mem(data);
|
||||
|
||||
ALOGI("Patterns offloaded successfully\n");
|
||||
}
|
||||
|
||||
int PatternOffload::sendData(uint8_t *data, int len)
|
||||
{
|
||||
int rc, status = 0;
|
||||
|
||||
if (!data || !len)
|
||||
return -EINVAL;
|
||||
|
||||
rc = GlinkCh.GlinkWrite(data, len);
|
||||
if (rc < 0)
|
||||
return rc;
|
||||
|
||||
rc = GlinkCh.GlinkPoll();
|
||||
if (rc < 0)
|
||||
return rc;
|
||||
|
||||
rc = GlinkCh.GlinkRead((uint8_t *)&status, 4);
|
||||
if (rc < 0)
|
||||
return rc;
|
||||
|
||||
if (status != OFFLOAD_SUCCESS)
|
||||
return -EIO;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
||||
int PatternOffload::initChannel()
|
||||
{
|
||||
std::string chname = "/dev/glinkpkt_slate_haptics_offload";
|
||||
int rc;
|
||||
|
||||
rc = GlinkCh.GlinkOpen(chname);
|
||||
if (rc < 0)
|
||||
{
|
||||
ALOGE("Failed to open Glink channel name %s\n", chname.c_str());
|
||||
return rc;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
#define GLINK_MAX_CONN_RETRIES 60
|
||||
int OffloadGlinkConnection::GlinkOpen(std::string& dev)
|
||||
{
|
||||
int tries = GLINK_MAX_CONN_RETRIES;
|
||||
dev_name = dev;
|
||||
|
||||
do {
|
||||
fd = ::open(dev_name.c_str(), O_RDWR);
|
||||
tries--;
|
||||
if (fd < 0)
|
||||
{
|
||||
ALOGE("%s: %s: open error(%s)", __func__, dev.c_str(), strerror(errno));
|
||||
sleep(1);
|
||||
}
|
||||
} while(-ETIMEDOUT == errno && tries > 0 );
|
||||
|
||||
return fd;
|
||||
}
|
||||
|
||||
int OffloadGlinkConnection::GlinkClose()
|
||||
{
|
||||
if (fd >= 0)
|
||||
{
|
||||
::close(fd);
|
||||
fd = -1;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
int OffloadGlinkConnection::GlinkPoll()
|
||||
{
|
||||
ssize_t rc = 0;
|
||||
struct pollfd poll_fd;
|
||||
|
||||
// wait for Rx data available in fd, for 2 seconds timeout
|
||||
poll_fd.fd = fd;
|
||||
poll_fd.events = POLLIN;
|
||||
|
||||
rc = ::poll(&poll_fd, 1, 2000);
|
||||
|
||||
if(rc > 0)
|
||||
{
|
||||
if (poll_fd.revents & POLLIN)
|
||||
return 0;
|
||||
} else if (rc == 0) {
|
||||
ALOGE("Glink poll timeout");
|
||||
} else {
|
||||
ALOGE("Glink poll error: %s\n", strerror(errno));
|
||||
}
|
||||
|
||||
return -1;
|
||||
}
|
||||
|
||||
int OffloadGlinkConnection::GlinkRead(uint8_t *data, size_t size)
|
||||
{
|
||||
int rc = 0;
|
||||
size_t bytes_read = 0;
|
||||
|
||||
if (fd < 0)
|
||||
return -1;
|
||||
|
||||
if (0 != GlinkPoll())
|
||||
return -1;
|
||||
|
||||
while (bytes_read < size)
|
||||
{
|
||||
rc = ::read(fd, data+bytes_read, size-bytes_read);
|
||||
if (rc < 0) {
|
||||
if (errno != EAGAIN) {
|
||||
ALOGE("%s: Read error: %s, rc %d", __func__,
|
||||
strerror(errno), rc);
|
||||
goto read_error;
|
||||
}
|
||||
} else if (rc == 0) {
|
||||
ALOGE("%s: Zero length packet received or hardware connection went off",
|
||||
__func__);
|
||||
}
|
||||
bytes_read += rc;
|
||||
}
|
||||
return 0;
|
||||
|
||||
read_error:
|
||||
return -1;
|
||||
}
|
||||
|
||||
int OffloadGlinkConnection::GlinkWrite(uint8_t *buf, size_t buflen)
|
||||
{
|
||||
size_t bytes_written_out = 0;
|
||||
int rc = 0;
|
||||
|
||||
if (fd < 0)
|
||||
return -1;
|
||||
|
||||
if (!buflen) {
|
||||
ALOGE("%s: Invalid buffer len", __func__);
|
||||
return 0;
|
||||
}
|
||||
|
||||
while (bytes_written_out < buflen) {
|
||||
rc = ::write (fd, buf+bytes_written_out, buflen-bytes_written_out);
|
||||
if (rc < 0) {
|
||||
ALOGE("%s: Write returned failure %d", __func__, rc);
|
||||
return errno;
|
||||
}
|
||||
bytes_written_out += rc;
|
||||
}
|
||||
rc = 0;
|
||||
|
||||
return rc;
|
||||
}
|
||||
|
||||
} // namespace vibrator
|
||||
} // namespace hardware
|
||||
} // namespace android
|
||||
} // namespace aidl
|
||||
Reference in New Issue
Block a user