/* * Copyright (C) 2016 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. */ #define LOG_TAG "VrHALImpl" #include <cutils/log.h> #include <errno.h> #include <fcntl.h> #include <string.h> #include <sys/stat.h> #include <sys/types.h> #include <hardware/vr.h> #include <hardware/hardware.h> // Angler has two inflight numbers. By default, inflight=15 and inflight_low_latency=4. // Inflight is only used when there is a single GL context, when there is more than one // context, inflight_low_latency is used. Since we are only interested in affecting // performance when there is context preemption, we only have to modify the low latency // parameter. static const int DEFAULT_GPU_INFLIGHT = 4; static const int VR_MODE_GPU_INFLIGHT = 2; static const char* GPU_INFLIGHT_PATH = "/sys/class/kgsl/kgsl-3d0/dispatch/inflight_low_latency"; /** * Write 'len' characters from 'input' character array into file at path 'outFile.' * * Return 0 on success, or a negative error code. */ static int write_string(const char* input, size_t len, const char* outFile) { int fd = -1; ssize_t err = 0; // Check input strings. if (input == NULL || outFile == NULL) { ALOGE("%s: Invalid input to write", __FUNCTION__); return -1; } // Open file, check for errors. fd = open(outFile, O_WRONLY); if (fd < 0) { ALOGE("%s: Failed to open file %s, error %s (%d)", __FUNCTION__, outFile, strerror(errno), -errno); return -errno; } // Write file, check for errors. err = write(fd, input, len); if (err < 0) { ALOGE("%s: Failed to write file %s, error %s (%d)", __FUNCTION__, outFile, strerror(errno), -errno); close(fd); return -errno; } // Close and return success. close(fd); return 0; } /** * Write integer 'input' formatted as a character string into the file at path 'outFile.' * * Return 0 on success, or a negative error code. */ static int write_int(int input, const char* outFile) { char buffer[128] = {0,}; int bytes = snprintf(buffer, sizeof(buffer), "%d", input); if (bytes < 0 || (size_t) bytes >= sizeof(buffer)) { ALOGE("%s: Failed to format integer %d", __FUNCTION__, input); return -EINVAL; } return write_string(buffer, (size_t) bytes, outFile); } // Set global display/GPU/scheduler configuration to used for VR apps. static void set_vr_performance_configuration() { int err = 0; // Set in-flight buffers to 2. err = write_int(VR_MODE_GPU_INFLIGHT, GPU_INFLIGHT_PATH); if (err < 0) { ALOGW("%s: Error while setting configuration for VR mode.", __FUNCTION__); } } // Reset to default global display/GPU/scheduler configuration. static void unset_vr_performance_configuration() { int err = 0; // Set in-flight buffers back to default (15). err = write_int(DEFAULT_GPU_INFLIGHT, GPU_INFLIGHT_PATH); if (err < 0) { ALOGW("%s: Error while setting configuration for VR mode.", __FUNCTION__); } } static void vr_init(struct vr_module *module) { // NOOP } static void vr_set_vr_mode(struct vr_module *module, bool enabled) { if (enabled) { set_vr_performance_configuration(); } else { unset_vr_performance_configuration(); } } static struct hw_module_methods_t vr_module_methods = { .open = NULL, // There are no devices for this HAL interface. }; vr_module_t HAL_MODULE_INFO_SYM = { .common = { .tag = HARDWARE_MODULE_TAG, .module_api_version = VR_MODULE_API_VERSION_1_0, .hal_api_version = HARDWARE_HAL_API_VERSION, .id = VR_HARDWARE_MODULE_ID, .name = "Angler VR HAL", .author = "The Android Open Source Project", .methods = &vr_module_methods, }, .init = vr_init, .set_vr_mode = vr_set_vr_mode, };