blob: b8308ce16f1697f5877a9e64950e639b450ae7c3 [file] [log] [blame]
/*
* Copyright (C) 2019 The Android Open Source Project *
* 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 <stdlib.h>
#include <charconv>
#include "utils.h"
#include "vibrator.h"
namespace android {
namespace idlcli {
class CommandVibrator;
namespace vibrator {
using aidl::ActivePwle;
using aidl::Braking;
using aidl::BrakingPwle;
using aidl::PrimitivePwle;
class CommandComposePwle : public Command {
std::string getDescription() const override { return "Compose PWLE vibration."; }
std::string getUsageSummary() const override {
return "[options] a <active pwle params> b <braking pwle params> ...";
}
UsageDetails getUsageDetails() const override {
UsageDetails details{
{"-b", {"Block for duration of vibration."}},
{"a <startAmplitude> <startFrequency> <endAmplitude> <endFrequency> <duration>",
{"Enter the active PWLE segment parameters"}},
{"b <brakingMethod> <duration>", {"Enter the braking PWLE segment parameters"}},
{"...", {"May repeat multiple times."}},
};
return details;
}
int getIntFromString(std::string input, int *output) {
int rc = 0;
int value;
const auto res = std::from_chars(input.data(), input.data() + input.size(), value);
if (res.ec == std::errc::invalid_argument) {
std::cerr << "Invalid int argument: " << input << std::endl;
rc = (int)std::errc::invalid_argument;
} else if (res.ec == std::errc::result_out_of_range) {
std::cerr << "Result out of range: " << input << std::endl;
rc = (int)std::errc::result_out_of_range;
}
*output = value;
return rc;
}
float getFloatFromString(std::string_view input, float *output) {
int rc = 0;
errno = 0;
// from_chars doesn't support conversion to float so we need to first
// convert the string_view to string and use the C-string for strtof
float value = strtof(std::string(input).c_str(), NULL);
if (input == "0.0" || input == "0") {
return rc;
}
if (value <= 0.0) {
std::cerr << "Invalid float argument: " << input << std::endl;
rc = EINVAL;
} else if (errno == ERANGE) {
std::cerr << "Result out of range: " << input << std::endl;
rc = errno;
} else {
*output = value;
}
return rc;
}
Status doArgs(Args &args) override {
while (args.get<std::string>().value_or("").find("-") == 0) {
auto opt = *args.pop<std::string>();
if (opt == "--") {
break;
} else if (opt == "-b") {
mBlocking = true;
} else {
std::cerr << "Invalid Option '" << opt << "'!" << std::endl;
return USAGE;
}
}
if (args.empty()) {
std::cerr << "Missing arguments! Please see usage" << std::endl;
return USAGE;
}
while (!args.empty()) {
PrimitivePwle pwle;
auto nextArg = args.pop();
if (*nextArg == "a") {
auto startAmplitude = args.pop();
float startAmp;
if (getFloatFromString(*startAmplitude, &startAmp))
return USAGE;
auto startFrequency = args.pop();
float startFreq;
if (getFloatFromString(*startFrequency, &startFreq))
return USAGE;
auto endAmplitude = args.pop();
float endAmp;
if (getFloatFromString(*endAmplitude, &endAmp))
return USAGE;
auto endFrequency = args.pop();
float endFreq;
if (getFloatFromString(*endFrequency, &endFreq))
return USAGE;
auto duration = args.pop();
int dur;
if (getIntFromString(*duration, &dur))
return USAGE;
ActivePwle active = {startAmp, startFreq, endAmp, endFreq, dur};
pwle = active;
} else if (*nextArg == "b") {
auto brakingMethod = args.pop();
Braking brakingMeth;
if (getIntFromString(*brakingMethod, (int *)&brakingMeth))
return USAGE;
auto duration = args.pop();
int dur;
if (getIntFromString(*duration, &dur))
return USAGE;
BrakingPwle braking = {brakingMeth, dur};
pwle = braking;
} else {
std::cerr << "Invalid arguments! Please see usage" << std::endl;
return USAGE;
}
mCompositePwle.emplace_back(std::move(pwle));
}
if (!args.empty()) {
std::cerr << "Unexpected Arguments!" << std::endl;
return USAGE;
}
return OK;
}
Status doMain(Args && /*args*/) override {
auto hal = getHal<aidl::IVibrator>();
if (!hal) {
return UNAVAILABLE;
}
ABinderProcess_setThreadPoolMaxThreadCount(1);
ABinderProcess_startThreadPool();
std::shared_ptr<VibratorCallback> callback;
if (mBlocking) {
callback = ndk::SharedRefBase::make<VibratorCallback>();
}
auto status = hal->call(&aidl::IVibrator::composePwle, mCompositePwle, callback);
if (status.isOk() && callback) {
callback->waitForComplete();
}
std::cout << "Status: " << status.getDescription() << std::endl;
return status.isOk() ? OK : ERROR;
}
bool mBlocking;
std::vector<PrimitivePwle> mCompositePwle;
};
static const auto Command =
CommandRegistry<CommandVibrator>::Register<CommandComposePwle>("composePwle");
} // namespace vibrator
} // namespace idlcli
} // namespace android