blob: 255a1986b2cdda1e9ac39bac02a6f68810621bce [file] [log] [blame]
// Copyright (C) 2020 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 "GfxStreamAgents.h"
#include <stdint.h> // for uint32_t
#include <stdio.h> // for fprintf
#include <map> // for map, __ma...
#include <utility> // for pair
#include "host-common/HostmemIdMapping.h" // for android_e...
#include "host-common/MultiDisplay.h" // for MultiDisp...
#include "host-common/multi_display_agent.h" // for QAndroidM...
#include "host-common/vm_operations.h" // for SnapshotC...
#include "host-common/window_agent.h" // for WindowMes...
#include "host-common/misc.h"
#ifdef _DEBUG
#define DEBUG_LOG(fd, fmt, ...) fprintf(fd, fmt, __VA_ARGS__);
#else
#define DEBUG_LOG(fd, fmt, ...)
#endif
static std::map<uint32_t, android::MultiDisplayInfo> mMultiDisplay;
using namespace android;
static const QAndroidMultiDisplayAgent sMultiDisplayAgent = {
.setMultiDisplay = [](uint32_t id,
int32_t x,
int32_t y,
uint32_t w,
uint32_t h,
uint32_t dpi,
uint32_t flag,
bool add) -> int {
return 0;
},
.getMultiDisplay = [](uint32_t id,
int32_t* x,
int32_t* y,
uint32_t* w,
uint32_t* h,
uint32_t* dpi,
uint32_t* flag,
bool* enabled) -> bool {
if (mMultiDisplay.find(id) == mMultiDisplay.end()) {
if (enabled) {
*enabled = false;
}
return false;
}
if (x) {
*x = mMultiDisplay[id].pos_x;
}
if (y) {
*y = mMultiDisplay[id].pos_y;
}
if (w) {
*w = mMultiDisplay[id].width;
}
if (h) {
*h = mMultiDisplay[id].height;
}
if (dpi) {
*dpi = mMultiDisplay[id].dpi;
}
if (flag) {
*flag = mMultiDisplay[id].flag;
}
if (enabled) {
*enabled = mMultiDisplay[id].enabled;
}
return true;
},
.getNextMultiDisplay = [](int32_t start_id,
uint32_t* id,
int32_t* x,
int32_t* y,
uint32_t* w,
uint32_t* h,
uint32_t* dpi,
uint32_t* flag,
uint32_t* cb) -> bool {
uint32_t key;
std::map<uint32_t, android::MultiDisplayInfo>::iterator i;
if (start_id < 0) {
key = 0;
} else {
key = start_id + 1;
}
i = mMultiDisplay.lower_bound(key);
if (i == mMultiDisplay.end()) {
return false;
} else {
if (id) {
*id = i->first;
}
if (x) {
*x = i->second.pos_x;
}
if (y) {
*y = i->second.pos_y;
}
if (w) {
*w = i->second.width;
}
if (h) {
*h = i->second.height;
}
if (dpi) {
*dpi = i->second.dpi;
}
if (flag) {
*flag = i->second.flag;
}
if (cb) {
*cb = i->second.cb;
}
return true;
}
},
.isMultiDisplayEnabled = [](void) -> bool {
return mMultiDisplay.size() > 1;
},
.getCombinedDisplaySize = [](uint32_t* width, uint32_t* height) {},
.multiDisplayParamValidate = [](uint32_t id,
uint32_t w,
uint32_t h,
uint32_t dpi,
uint32_t flag) -> bool { return true; },
.translateCoordination =
[](uint32_t* x, uint32_t* y, uint32_t* displayId) -> bool {
return true;
},
.setGpuMode = [](bool isGuestMode, uint32_t w, uint32_t h) {},
.createDisplay = [](uint32_t* displayId) -> int {
if (displayId == nullptr) {
fprintf(stderr, "null displayId pointer\n");
return -1;
}
if (mMultiDisplay.size() >= MultiDisplay::s_maxNumMultiDisplay) {
fprintf(stderr, "cannot create more displays, exceeding limits %d\n",
MultiDisplay::s_maxNumMultiDisplay);
return -1;
}
if (mMultiDisplay.find(*displayId) != mMultiDisplay.end()) {
return 0;
}
// displays created by internal rcCommands
if (*displayId == MultiDisplay::s_invalidIdMultiDisplay) {
for (int i = MultiDisplay::s_displayIdInternalBegin; i < MultiDisplay::s_maxNumMultiDisplay; i++) {
if (mMultiDisplay.find(i) == mMultiDisplay.end()) {
*displayId = i;
break;
}
}
}
if (*displayId == MultiDisplay::s_invalidIdMultiDisplay) {
fprintf(stderr, "cannot create more internaldisplays, exceeding limits %d\n",
MultiDisplay::s_maxNumMultiDisplay - MultiDisplay::s_displayIdInternalBegin);
return -1;
}
mMultiDisplay.emplace(*displayId, android::MultiDisplayInfo());
return 0;
},
.destroyDisplay = [](uint32_t displayId) -> int {
mMultiDisplay.erase(displayId);
return 0;
},
.setDisplayPose = [](uint32_t displayId,
int32_t x,
int32_t y,
uint32_t w,
uint32_t h,
uint32_t dpi) -> int {
if (mMultiDisplay.find(displayId) == mMultiDisplay.end()) {
fprintf(stderr, "cannot find display %d\n", displayId);
return -1;
}
mMultiDisplay[displayId].pos_x = x;
mMultiDisplay[displayId].pos_y = y;
mMultiDisplay[displayId].width = w;
mMultiDisplay[displayId].height = h;
mMultiDisplay[displayId].dpi = dpi;
return 0;
},
.getDisplayPose = [](uint32_t displayId,
int32_t* x,
int32_t* y,
uint32_t* w,
uint32_t* h) -> int {
if (mMultiDisplay.find(displayId) == mMultiDisplay.end()) {
fprintf(stderr, "cannot find display %d\n", displayId);
return -1;
}
if (x)
*x = mMultiDisplay[displayId].pos_x;
if (y)
*y = mMultiDisplay[displayId].pos_y;
if (w)
*w = mMultiDisplay[displayId].width;
if (h)
*h = mMultiDisplay[displayId].height;
return 0;
},
.getDisplayColorBuffer = [](uint32_t displayId,
uint32_t* colorBuffer) -> int {
if (mMultiDisplay.find(displayId) == mMultiDisplay.end()) {
fprintf(stderr, "cannot find display %d\n", displayId);
return -1;
}
*colorBuffer = mMultiDisplay[displayId].cb;
return 0;
},
.getColorBufferDisplay = [](uint32_t colorBuffer,
uint32_t* displayId) -> int {
for (const auto& iter : mMultiDisplay) {
if (iter.second.cb == colorBuffer) {
*displayId = iter.first;
return 0;
}
}
return -1;
},
.setDisplayColorBuffer = [](uint32_t displayId,
uint32_t colorBuffer) -> int {
if (mMultiDisplay.find(displayId) == mMultiDisplay.end()) {
fprintf(stderr, "cannot find display %d\n", displayId);
return -1;
}
mMultiDisplay[displayId].cb = colorBuffer;
return 0;
},
.isPixelFold = [] () {
return false;
},
};
static bool sIsFolded = false;
static const QAndroidEmulatorWindowAgent sQAndroidEmulatorWindowAgent = {
.getEmulatorWindow =
[](void) {
DEBUG_LOG(stderr,
"window-agent-GfxStream-impl: "
".getEmulatorWindow\n");
return (EmulatorWindow*)nullptr;
},
.rotate90Clockwise =
[](void) {
DEBUG_LOG(stderr,
"window-agent-GfxStream-impl: "
".rotate90Clockwise\n");
return true;
},
.rotate =
[](int rotation) {
DEBUG_LOG(stderr,
"window-agent-GfxStream-impl: "
".rotate90Clockwise\n");
return true;
},
.getRotation =
[](void) {
DEBUG_LOG(stderr,
"window-agent-GfxStream-impl: .getRotation\n");
return (int)SKIN_ROTATION_0;
},
.showMessage =
[](const char* message, WindowMessageType type, int timeoutMs) {
DEBUG_LOG(stderr,
"window-agent-GfxStream-impl: .showMessage %s\n",
message);
},
.showMessageWithDismissCallback =
[](const char* message,
WindowMessageType type,
const char* dismissText,
void* context,
void (*func)(void*),
int timeoutMs) {
DEBUG_LOG(stderr,
"window-agent-GfxStream-impl: "
".showMessageWithDismissCallback %s\n",
message);
},
.fold =
[](bool is_fold) -> bool {
DEBUG_LOG(stderr, "window-agent-GfxStream-impl: .fold %d\n",
is_fold);
sIsFolded = is_fold;
return true;
},
.isFolded = [](void) -> bool { return sIsFolded; },
.getFoldedArea = [](int* x, int* y, int* w, int* h) -> bool {
DEBUG_LOG(stderr, "window-agent-GfxStream-impl: .getFoldedArea\n");
return true;
},
.updateFoldablePostureIndicator = [](bool) {
DEBUG_LOG(stderr, "window-agent-GfxStream-impl: updateFoldablePostureIndicator\n");
},
.setUIDisplayRegion =
[](int x_offset, int y_offset, int w, int h, bool ignoreOrientation) {
DEBUG_LOG(stderr,
"window-agent-GfxStream-impl: .setUIDisplayRegion "
"%d %d %dx%d\n",
x_offset, y_offset, w, h);
},
.getMultiDisplay = 0,
.setNoSkin = [](void) {},
.restoreSkin = [](void) {},
.updateUIMultiDisplayPage =
[](uint32_t id) {
DEBUG_LOG(stderr, "updateMultiDisplayPage\n");
},
.getMonitorRect =
[](uint32_t* w, uint32_t* h) -> bool {
if (w)
*w = 2500;
if (h)
*h = 1600;
return true;
},
};
static const QAndroidVmOperations sQAndroidVmOperations =
{
.vmStop = []() -> bool {
DEBUG_LOG(stderr, "goldfish-opengl vm ops: vm stop\n");
return true;
},
.vmStart = []() -> bool {
DEBUG_LOG(stderr, "goldfish-opengl vm ops: vm start\n");
return true;
},
.vmReset = []() { DEBUG_LOG(stderr, "goldfish-opengl vm ops: vm reset\n"); },
.vmShutdown = []() { DEBUG_LOG(stderr, "goldfish-opengl vm ops: vm reset\n"); },
.vmPause = []() -> bool {
DEBUG_LOG(stderr, "goldfish-opengl vm ops: vm pause\n");
return true;
},
.vmResume = []() -> bool {
DEBUG_LOG(stderr, "goldfish-opengl vm ops: vm resume\n");
return true;
},
.vmIsRunning = []() -> bool {
DEBUG_LOG(stderr, "goldfish-opengl vm ops: vm is running\n");
return true;
},
.snapshotList = [](void*, LineConsumerCallback, LineConsumerCallback) -> bool {
DEBUG_LOG(stderr, "goldfish-opengl vm ops: snapshot list\n");
return true;
},
.snapshotSave = [](const char* name, void* opaque, LineConsumerCallback) -> bool {
DEBUG_LOG(stderr, "gfxstream vm ops: snapshot save\n");
return true;
},
.snapshotLoad = [](const char* name, void* opaque, LineConsumerCallback) -> bool {
DEBUG_LOG(stderr, "gfxstream vm ops: snapshot load\n");
return true;
},
.snapshotDelete = [](const char* name, void* opaque,
LineConsumerCallback errConsumer) -> bool {
DEBUG_LOG(stderr, "goldfish-opengl vm ops: snapshot delete\n");
return true;
},
.snapshotRemap = [](bool shared, void* opaque, LineConsumerCallback errConsumer) -> bool {
DEBUG_LOG(stderr, "goldfish-opengl vm ops: snapshot remap\n");
return true;
},
.snapshotExport = [](const char* snapshot, const char* dest, void* opaque,
LineConsumerCallback errConsumer) -> bool {
DEBUG_LOG(stderr, "goldfish-opengl vm ops: snapshot export image\n");
return true;
},
.setSnapshotCallbacks =
[](void* opaque, const SnapshotCallbacks* callbacks) {
DEBUG_LOG(stderr, "goldfish-opengl vm ops: set snapshot callbacks\n");
},
.mapUserBackedRam =
[](uint64_t gpa, void* hva, uint64_t size) {
DEBUG_LOG(stderr, "%s: map user backed ram\n", __func__);
},
.unmapUserBackedRam =
[](uint64_t gpa, uint64_t size) {
DEBUG_LOG(stderr, "%s: unmap user backed ram\n", __func__);
},
.getVmConfiguration =
[](VmConfiguration* out) {
DEBUG_LOG(stderr, "goldfish-opengl vm ops: get vm configuration\n");
},
.setFailureReason =
[](const char* name, int failureReason) {
DEBUG_LOG(stderr, "goldfish-opengl vm ops: set failure reason\n");
},
.setExiting = []() { DEBUG_LOG(stderr, "goldfish-opengl vm ops: set exiting\n"); },
.allowRealAudio =
[](bool allow) { DEBUG_LOG(stderr, "goldfish-opengl vm ops: allow real audio\n"); },
.physicalMemoryGetAddr =
[](uint64_t gpa) {
DEBUG_LOG(stderr, "%s: physmemGetAddr\n", __func__);
return (void*)nullptr;
},
.isRealAudioAllowed =
[](void) {
DEBUG_LOG(stderr, "goldfish-opengl vm ops: is real audiop allowed\n");
return true;
},
.setSkipSnapshotSave =
[](bool used) {
DEBUG_LOG(stderr, "goldfish-opengl vm ops: set skip snapshot save\n");
},
.isSnapshotSaveSkipped =
[]() {
DEBUG_LOG(stderr,
"goldfish-opengl vm ops: is snapshot save "
"skipped\n");
return false;
},
.hostmemRegister = android_emulation_hostmem_register,
.hostmemUnregister = android_emulation_hostmem_unregister,
.hostmemGetInfo = android_emulation_hostmem_get_info,
#ifdef GFXSTREAM_ENABLE_HOST_VK_SNAPSHOT
.setSkipSnapshotSaveReason =
[](SnapshotSkipReason reason) {
DEBUG_LOG(stderr,
"goldfish-opengl vm ops: set skip snapshot save reason"
"skipped\n");
},
.getSkipSnapshotSaveReason =
[]() {
DEBUG_LOG(stderr,
"goldfish-opengl vm ops: get skip snapshot save "
"reason\n");
return SNAPSHOT_SKIP_UNKNOWN;
},
.setStatSnapshotUseVulkan =
[]() {
DEBUG_LOG(stderr,
"goldfish-opengl vm ops: set stat snapshot use Vulkan"
"skipped\n");
},
.snapshotUseVulkan =
[]() {
DEBUG_LOG(stderr,
"goldfish-opengl vm ops: get stat snapshot use Vulkan"
"skipped\n");
return false;
},
#endif
};
namespace android {
namespace emulation {
const QAndroidVmOperations* const
GfxStreamGraphicsAgentFactory::android_get_QAndroidVmOperations() const {
return &sQAndroidVmOperations;
}
const QAndroidMultiDisplayAgent* const
GfxStreamGraphicsAgentFactory::android_get_QAndroidMultiDisplayAgent() const {
return &sMultiDisplayAgent;
}
const QAndroidEmulatorWindowAgent* const
GfxStreamGraphicsAgentFactory::android_get_QAndroidEmulatorWindowAgent()
const {
return &sQAndroidEmulatorWindowAgent;
}
} // namespace emulation
} // namespace android