xrexpirament/main.c
2025-04-15 21:13:31 +02:00

195 lines
6.1 KiB
C

#include <stdint.h>
#include <stdio.h>
#include <openxr/openxr.h>
#include <stdlib.h>
#include <stdbool.h>
#include <string.h>
#include <signal.h>
#include <time.h>
sig_atomic_t stop = false;
void handler(int sig)
{
stop = true;
}
#define HEADLESS_EXT "XR_MND_headless"
static const char* xr_strerror(XrInstance instance, XrResult err)
{
static char buffer[XR_MAX_RESULT_STRING_SIZE];
xrResultToString(instance, err, buffer);
return buffer;
}
void clear()
{
puts("\033[H\033[J");
}
int main(int argc, char** argv)
{
XrInstance instance;
XrApplicationInfo applicationInfo = {
.applicationName = "OpenXR Example",
.engineName = "Example Engine",
.apiVersion = XR_API_VERSION_1_0,
};
uint32_t extentionPropCount;
xrEnumerateInstanceExtensionProperties(NULL, 0, &extentionPropCount, NULL);
XrExtensionProperties *proparties = calloc(extentionPropCount, sizeof(*proparties));
for (uint32_t i = 0; i < extentionPropCount; ++i)
proparties[i].type = XR_TYPE_EXTENSION_PROPERTIES;
XrResult result = xrEnumerateInstanceExtensionProperties(NULL, extentionPropCount, &extentionPropCount, proparties);
if (!XR_SUCCEEDED(result)) {
printf("Failed to query OpenXR extensions: %d\n", result);
return 1;
}
printf("OpenXR extensions: %d\n", extentionPropCount);
bool foundHeadless = false;
for (uint32_t i = 0; i < extentionPropCount; ++i) {
printf("\t%s\n", proparties[i].extensionName);
if (strcmp(HEADLESS_EXT, proparties[i].extensionName) == 0)
foundHeadless = true;
}
if (!foundHeadless) {
puts("Runtime dose not support " HEADLESS_EXT);
return 1;
}
const char *enabledExtensionNames[] = {HEADLESS_EXT, NULL};
XrInstanceCreateInfo createInfo = {
.type = XR_TYPE_INSTANCE_CREATE_INFO,
.next = NULL,
.createFlags = 0,
.applicationInfo = applicationInfo,
.enabledExtensionNames = enabledExtensionNames,
.enabledExtensionCount = 1
};
XrSystemGetInfo getInfo = {
.type = XR_TYPE_SYSTEM_GET_INFO,
.formFactor = XR_FORM_FACTOR_HEAD_MOUNTED_DISPLAY
};
result = xrCreateInstance(&createInfo, &instance);
if (XR_SUCCEEDED(result)) {
printf("OpenXR instance created successfully.\n");
} else {
printf("Failed to create OpenXR instance: %d\n", result);
return 1;
}
XrSystemId systemId;
result = xrGetSystem(instance, &getInfo, &systemId);
if (!XR_SUCCEEDED(result)) {
printf("Failed to get OpenXR system: %s\n", xr_strerror(instance, result));
return 1;
}
uint32_t viewCount;
result = xrEnumerateViewConfigurationViews(instance, systemId, XR_VIEW_CONFIGURATION_TYPE_PRIMARY_STEREO, 0, &viewCount, NULL);
XrViewConfigurationView* viewConfigs = calloc(viewCount, sizeof(XrViewConfigurationView));
for (size_t i = 0; i < viewCount; ++i)
viewConfigs[i].type = XR_TYPE_VIEW_CONFIGURATION_VIEW;
if (XR_SUCCEEDED(result)) {
result = xrEnumerateViewConfigurationViews(instance, systemId, XR_VIEW_CONFIGURATION_TYPE_PRIMARY_STEREO, viewCount, &viewCount, viewConfigs);
if (!XR_SUCCEEDED(result)) {
printf("Failed to enumerate viewConfigs: %d\n", result);
return 1;
}
}
XrSession session;
XrSessionCreateInfo createSessionInfo = {
.type = XR_TYPE_SESSION_CREATE_INFO,
.systemId = systemId
};
result = xrCreateSession(instance, &createSessionInfo, &session);
if (!XR_SUCCEEDED(result)) {
printf("Failed to create session: %s\n", xr_strerror(instance, result));
return 1;
}
XrSessionBeginInfo beginInfo = {.type = XR_TYPE_SESSION_BEGIN_INFO, .primaryViewConfigurationType = XR_VIEW_CONFIGURATION_TYPE_PRIMARY_STEREO};
result = xrBeginSession(session, &beginInfo);
XrSpace stageSpace;
XrReferenceSpaceCreateInfo spaceCreateInfo = {
.type = XR_TYPE_REFERENCE_SPACE_CREATE_INFO,
.referenceSpaceType = XR_REFERENCE_SPACE_TYPE_STAGE
};
spaceCreateInfo.poseInReferenceSpace.orientation.x = 1;
result = xrCreateReferenceSpace(session, &spaceCreateInfo, &stageSpace);
if (!XR_SUCCEEDED(result)) {
printf("Failed to create referance space: %s\n", xr_strerror(instance, result));
return 1;
}
struct timespec duration = {.tv_sec = 1};
nanosleep(&duration, NULL);
XrView *views = calloc(viewCount, sizeof(*views));
for (size_t i = 0; i < viewCount; ++i)
views[i].type = XR_TYPE_VIEW;
XrViewState state = {.type = XR_TYPE_VIEW_STATE};
signal(SIGTERM, handler);
signal(SIGHUP, handler);
signal(SIGINT, handler);
while(!stop) {
XrFrameWaitInfo waitInfo = {.type = XR_TYPE_FRAME_WAIT_INFO};
XrFrameState frameState = {.type = XR_TYPE_FRAME_STATE};
result = xrWaitFrame(session, &waitInfo, &frameState);
if (!XR_SUCCEEDED(result)) {
printf("Failed to wait for frame: %s\n", xr_strerror(instance, result));
return 1;
} else {
printf("waited for frame time: %lu\n", frameState.predictedDisplayPeriod);
}
XrViewLocateInfo locateInfo = {
.type = XR_TYPE_VIEW_LOCATE_INFO,
.viewConfigurationType = XR_VIEW_CONFIGURATION_TYPE_PRIMARY_STEREO,
.space = stageSpace,
.displayTime = frameState.predictedDisplayTime+1
};
uint32_t locatedViews;
result = xrLocateViews(session, &locateInfo, &state, viewCount, &locatedViews, views);
if (!XR_SUCCEEDED(result)) {
printf("Failed to locate views: %s\n", xr_strerror(instance, result));
struct timespec duration = {.tv_nsec = 1000*100};
nanosleep(&duration, NULL);
continue;
} else if (locatedViews != viewCount) {
printf("Failed to locate all views");
struct timespec duration = {.tv_nsec = 1000*100};
nanosleep(&duration, NULL);
continue;
}
clear();
for (size_t i = 0; i < viewCount; ++i) {
printf("View %zu:\n", i);
printf("\tRecommended Size: %d x %d\n", viewConfigs[i].recommendedImageRectWidth, viewConfigs[i].recommendedImageRectHeight);
printf("\tFOV:\n\t\tUP: %f\n\t\tDOWN: %f\n\t\tLEFT: %f\n\t\tRIGHT: %f\n", views[i].fov.angleUp, views[i].fov.angleDown, views[i].fov.angleLeft, views[i].fov.angleRight);
printf("\tPOS:\n\t\tX: %f\n\t\tY: %f\n\t\tZ: %f\n", views[i].pose.position.x, views[i].pose.position.y, views[i].pose.position.z);
}
struct timespec duration = {.tv_nsec = 1000*100};
nanosleep(&duration, NULL);
}
xrEndSession(session);
free(views);
free(viewConfigs);
xrDestroySpace(stageSpace);
xrDestroySession(session);
xrDestroyInstance(instance);
return 0;
}