diff --git a/build.sh b/build.sh index 0c3f6f95bda..d73fbeb61b1 100755 --- a/build.sh +++ b/build.sh @@ -234,6 +234,7 @@ case "$platform" in # Adjust compilation options based on platform CYGWIN*|MINGW*|MSYS*) echo 'Compiling for Windows...' sys_cflags='-Wno-error' + CFLAGS="${CFLAGS} -lIphlpapi -lCrypt32" # workaround for linking libs on mingw opts="$opts --disable-fortify-source" postbuild='package_windows' # set the above function to be called after build target="qemu-system-i386w.exe" diff --git a/config_spec.yml b/config_spec.yml index b609bf859e5..b858606e685 100644 --- a/config_spec.yml +++ b/config_spec.yml @@ -25,6 +25,27 @@ input: port2: string port3: string port4: string + peripherals: + port1: + peripheral_type_0: integer + peripheral_param_0: string + peripheral_type_1: integer + peripheral_param_1: string + port2: + peripheral_type_0: integer + peripheral_param_0: string + peripheral_type_1: integer + peripheral_param_1: string + port3: + peripheral_type_0: integer + peripheral_param_0: string + peripheral_type_1: integer + peripheral_param_1: string + port4: + peripheral_type_0: integer + peripheral_param_0: string + peripheral_type_1: integer + peripheral_param_1: string gamecontrollerdb_path: string auto_bind: type: bool @@ -156,6 +177,26 @@ display: auto_scale: type: bool default: true + debug: + video: + transparency: + type: bool + default: false + x_pos: + type: number + default: 100.0 + y_pos: + type: number + default: 100.0 + x_winsize: + type: number + default: 600.0 + y_winsize: + type: number + default: 150.0 + advanced_tree_state: + type: bool + default: false audio: use_dsp: bool diff --git a/data/meson.build b/data/meson.build index 89e6123d90d..e1a7ebedf4e 100644 --- a/data/meson.build +++ b/data/meson.build @@ -1,5 +1,6 @@ pfiles = [ 'controller_mask.png', + 'xmu_mask.png', 'logo_sdf.png', 'xemu_64x64.png', 'abxy.ttf', diff --git a/data/xmu_mask.png b/data/xmu_mask.png new file mode 100644 index 00000000000..1ff5a7489ed Binary files /dev/null and b/data/xmu_mask.png differ diff --git a/hw/xbox/nv2a/pgraph.c b/hw/xbox/nv2a/pgraph.c index 9d9247fecac..fbf58fce9ad 100644 --- a/hw/xbox/nv2a/pgraph.c +++ b/hw/xbox/nv2a/pgraph.c @@ -28,6 +28,9 @@ #include "ui/xemu-settings.h" #include "qemu/fast-hash.h" +const float f16_max = 511.9375f; +const float f24_max = 1.0E30; + static NV2AState *g_nv2a; GloContext *g_nv2a_context_render; GloContext *g_nv2a_context_display; @@ -2942,6 +2945,10 @@ DEF_METHOD(NV097, SET_BEGIN_END) glDisable(GL_CULL_FACE); } + /* Clipping */ + glEnable(GL_CLIP_DISTANCE0); + glEnable(GL_CLIP_DISTANCE1); + /* Front-face select */ glFrontFace(pg->regs[NV_PGRAPH_SETUPRASTER] & NV_PGRAPH_SETUPRASTER_FRONTFACE @@ -3493,10 +3500,6 @@ DEF_METHOD(NV097, CLEAR_SURFACE) GLint gl_clear_stencil; GLfloat gl_clear_depth; - /* FIXME: Put these in some lookup table */ - const float f16_max = 511.9375f; - const float f24_max = 1.0E30; - switch(pg->surface_shape.zeta_format) { case NV097_SET_SURFACE_FORMAT_ZETA_Z16: { uint16_t z = clear_zstencil & 0xFFFF; @@ -4180,8 +4183,17 @@ static void pgraph_shader_update_constants(PGRAPHState *pg, *(float*)&pg->regs[NV_PGRAPH_FOGPARAM1]); } - float zclip_max = *(float*)&pg->regs[NV_PGRAPH_ZCLIPMAX]; - float zclip_min = *(float*)&pg->regs[NV_PGRAPH_ZCLIPMIN]; + float zmax; + switch (pg->surface_shape.zeta_format) { + case NV097_SET_SURFACE_FORMAT_ZETA_Z16: + zmax = pg->surface_shape.z_format ? f16_max : (float)0xFFFF; + break; + case NV097_SET_SURFACE_FORMAT_ZETA_Z24S8: + zmax = pg->surface_shape.z_format ? f24_max : (float)0xFFFFFF; + break; + default: + assert(0); + } if (fixed_function) { /* update lighting constants */ @@ -4238,19 +4250,15 @@ static void pgraph_shader_update_constants(PGRAPHState *pg, float m11 = 0.5 * (pg->surface_binding_dim.width/aa_width); float m22 = -0.5 * (pg->surface_binding_dim.height/aa_height); - float m33 = zclip_max - zclip_min; + float m33 = zmax; float m41 = *(float*)&pg->vsh_constants[NV_IGRAPH_XF_XFCTX_VPOFF][0]; float m42 = *(float*)&pg->vsh_constants[NV_IGRAPH_XF_XFCTX_VPOFF][1]; - float m43 = zclip_min; - if (m33 == 0.0) { - m33 = 1.0; - } float invViewport[16] = { 1.0/m11, 0, 0, 0, 0, 1.0/m22, 0, 0, 0, 0, 1.0/m33, 0, - -1.0+m41/m11, 1.0+m42/m22, -m43/m33, 1.0 + -1.0+m41/m11, 1.0+m42/m22, 0, 1.0 }; if (binding->inv_viewport_loc != -1) { @@ -4284,7 +4292,9 @@ static void pgraph_shader_update_constants(PGRAPHState *pg, } if (binding->clip_range_loc != -1) { - glUniform2f(binding->clip_range_loc, zclip_min, zclip_max); + float zclip_min = *(float*)&pg->regs[NV_PGRAPH_ZCLIPMIN] / zmax * 2.0 - 1.0; + float zclip_max = *(float*)&pg->regs[NV_PGRAPH_ZCLIPMAX] / zmax * 2.0 - 1.0; + glUniform4f(binding->clip_range_loc, 0, zmax, zclip_min, zclip_max); } /* Clipping regions */ diff --git a/hw/xbox/nv2a/shaders.c b/hw/xbox/nv2a/shaders.c index d0a1c0fa7ca..cafe326e93e 100644 --- a/hw/xbox/nv2a/shaders.c +++ b/hw/xbox/nv2a/shaders.c @@ -267,6 +267,8 @@ static MString* generate_geometry_shader( "void emit_vertex(int index, int _unused) {\n" " gl_Position = gl_in[index].gl_Position;\n" " gl_PointSize = gl_in[index].gl_PointSize;\n" + " gl_ClipDistance[0] = gl_in[index].gl_ClipDistance[0];\n" + " gl_ClipDistance[1] = gl_in[index].gl_ClipDistance[1];\n" " vtx_inv_w = v_vtx_inv_w[index];\n" " vtx_inv_w_flat = v_vtx_inv_w[index];\n" " vtxD0 = v_vtxD0[index];\n" @@ -289,6 +291,8 @@ static MString* generate_geometry_shader( "void emit_vertex(int index, int provoking_index) {\n" " gl_Position = gl_in[index].gl_Position;\n" " gl_PointSize = gl_in[index].gl_PointSize;\n" + " gl_ClipDistance[0] = gl_in[index].gl_ClipDistance[0];\n" + " gl_ClipDistance[1] = gl_in[index].gl_ClipDistance[1];\n" " vtx_inv_w = v_vtx_inv_w[index];\n" " vtx_inv_w_flat = v_vtx_inv_w[provoking_index];\n" " vtxD0 = v_vtxD0[provoking_index];\n" @@ -784,7 +788,7 @@ static MString *generate_vertex_shader(const ShaderState *state, MString *header = mstring_from_str( "#version 400\n" "\n" -"uniform vec2 clipRange;\n" +"uniform vec4 clipRange;\n" "uniform vec2 surfaceSize;\n" "\n" /* All constants in 1 array declaration */ @@ -864,7 +868,6 @@ GLSL_DEFINE(texMat3, GLSL_C_MAT4(NV_IGRAPH_XF_XFCTX_T3MAT)) if (state->fixed_function) { generate_fixed_function(state, header, body); - } else if (state->vertex_program) { vsh_translate(VSH_VERSION_XVS, (uint32_t*)state->program_data, @@ -973,6 +976,8 @@ GLSL_DEFINE(texMat3, GLSL_C_MAT4(NV_IGRAPH_XF_XFCTX_T3MAT)) " vtxT3 = oT3 * vtx_inv_w;\n" " gl_Position = oPos;\n" " gl_PointSize = oPts.x;\n" + " gl_ClipDistance[0] = oPos.z - oPos.w*clipRange.z;\n" // Near + " gl_ClipDistance[1] = oPos.w*clipRange.w - oPos.z;\n" // Far "\n" "}\n", shade_model_mult, diff --git a/hw/xbox/nv2a/vsh.c b/hw/xbox/nv2a/vsh.c index 5f338e7368c..0e4cf314bc2 100644 --- a/hw/xbox/nv2a/vsh.c +++ b/hw/xbox/nv2a/vsh.c @@ -849,11 +849,6 @@ void vsh_translate(uint16_t version, mstring_append(body, " oPos.z = oPos.w;\n"); } mstring_append(body, - /* Map the clip range into clip space so z is clipped correctly. - * Note this makes the values in the depth buffer wrong. This should be - * handled with gl_ClipDistance instead, but that has performance issues - * on OS X. - */ " if (clipRange.y != clipRange.x) {\n" " oPos.z = (oPos.z - clipRange.x)/(0.5*(clipRange.y - clipRange.x)) - 1;\n" " }\n" diff --git a/hw/xbox/nvnet.c b/hw/xbox/nvnet.c index b45295d0c1f..2440b2cd93b 100644 --- a/hw/xbox/nvnet.c +++ b/hw/xbox/nvnet.c @@ -25,6 +25,7 @@ #include "hw/pci/pci.h" #include "hw/qdev-properties.h" #include "net/net.h" +#include "qemu/bswap.h" #include "qemu/iov.h" #include "migration/vmstate.h" #include "nvnet_regs.h" @@ -32,6 +33,8 @@ #define IOPORT_SIZE 0x8 #define MMIO_SIZE 0x400 +static const uint8_t bcast[] = { 0xff, 0xff, 0xff, 0xff, 0xff, 0xff }; + // #define DEBUG #ifdef DEBUG # define NVNET_DPRINTF(format, ...) printf(format, ## __VA_ARGS__) @@ -432,6 +435,58 @@ static bool nvnet_is_packet_oversized(size_t size) return size > RX_ALLOC_BUFSIZE; } +static bool receive_filter(NvNetState *s, const uint8_t *buf, int size) +{ + if (size < 6) { + return false; + } + + uint32_t rctl = nvnet_get_reg(s, NvRegPacketFilterFlags, 4); + int isbcast = !memcmp(buf, bcast, sizeof bcast); + + /* Broadcast */ + if (isbcast) { + /* FIXME: bcast filtering */ + trace_nvnet_rx_filter_bcast_match(); + return true; + } + + if (!(rctl & NVREG_PFF_MYADDR)) { + /* FIXME: Confirm PFF_MYADDR filters mcast */ + return true; + } + + /* Multicast */ + uint32_t addr[2]; + addr[0] = cpu_to_le32(nvnet_get_reg(s, NvRegMulticastAddrA, 4)); + addr[1] = cpu_to_le32(nvnet_get_reg(s, NvRegMulticastAddrB, 4)); + if (memcmp(addr, bcast, sizeof bcast)) { + uint32_t dest_addr[2]; + memcpy(dest_addr, buf, 6); + dest_addr[0] &= cpu_to_le32(nvnet_get_reg(s, NvRegMulticastMaskA, 4)); + dest_addr[1] &= cpu_to_le32(nvnet_get_reg(s, NvRegMulticastMaskB, 4)); + + if (!memcmp(dest_addr, addr, 6)) { + trace_nvnet_rx_filter_mcast_match(MAC_ARG(dest_addr)); + return true; + } else { + trace_nvnet_rx_filter_mcast_mismatch(MAC_ARG(dest_addr)); + } + } + + /* Unicast */ + addr[0] = cpu_to_le32(nvnet_get_reg(s, NvRegMacAddrA, 4)); + addr[1] = cpu_to_le32(nvnet_get_reg(s, NvRegMacAddrB, 4)); + if (!memcmp(buf, addr, 6)) { + trace_nvnet_rx_filter_ucast_match(MAC_ARG(buf)); + return true; + } else { + trace_nvnet_rx_filter_ucast_mismatch(MAC_ARG(buf)); + } + + return false; +} + static ssize_t nvnet_receive_iov(NetClientState *nc, const struct iovec *iov, int iovcnt) { @@ -443,10 +498,17 @@ static ssize_t nvnet_receive_iov(NetClientState *nc, if (nvnet_is_packet_oversized(size)) { /* Drop */ NVNET_DPRINTF("%s packet too large!\n", __func__); + trace_nvnet_rx_oversized(size); return size; } iov_to_buf(iov, iovcnt, 0, s->rx_dma_buf, size); + + if (!receive_filter(s, s->rx_dma_buf, size)) { + trace_nvnet_rx_filter_dropped(); + return size; + } + #ifdef DEBUG nvnet_hex_dump(s, s->rx_dma_buf, size); #endif @@ -472,7 +534,7 @@ static ssize_t nvnet_dma_packet_to_guest(NvNetState *s, dma_addr_t rx_ring_addr = nvnet_get_reg(s, NvRegRxRingPhysAddr, 4); rx_ring_addr += s->rx_ring_index * sizeof(desc); pci_dma_read(d, rx_ring_addr, &desc, sizeof(desc)); - NVNET_DPRINTF("RX: Looking at ring descriptor %d (0x%llx): ", + NVNET_DPRINTF("RX: Looking at ring descriptor %d (0x%" HWADDR_PRIx "): ", s->rx_ring_index, rx_ring_addr); NVNET_DPRINTF("Buffer: 0x%x, ", desc.packet_buffer); NVNET_DPRINTF("Length: 0x%x, ", desc.length); @@ -539,7 +601,7 @@ static ssize_t nvnet_dma_packet_from_guest(NvNetState *s) dma_addr_t tx_ring_addr = nvnet_get_reg(s, NvRegTxRingPhysAddr, 4); tx_ring_addr += s->tx_ring_index * sizeof(desc); pci_dma_read(d, tx_ring_addr, &desc, sizeof(desc)); - NVNET_DPRINTF("TX: Looking at ring desc %d (%llx): ", + NVNET_DPRINTF("TX: Looking at ring desc %d (%" HWADDR_PRIx "): ", s->tx_ring_index, tx_ring_addr); NVNET_DPRINTF("Buffer: 0x%x, ", desc.packet_buffer); NVNET_DPRINTF("Length: 0x%x, ", desc.length); @@ -847,7 +909,7 @@ static void nvnet_dump_ring_descriptors(NvNetState *s) dma_addr_t tx_ring_addr = nvnet_get_reg(s, NvRegTxRingPhysAddr, 4); tx_ring_addr += i * sizeof(desc); pci_dma_read(d, tx_ring_addr, &desc, sizeof(desc)); - NVNET_DPRINTF("TX: Dumping ring desc %d (%llx): ", + NVNET_DPRINTF("TX: Dumping ring desc %d (%" HWADDR_PRIx "): ", i, tx_ring_addr); NVNET_DPRINTF("Buffer: 0x%x, ", desc.packet_buffer); NVNET_DPRINTF("Length: 0x%x, ", desc.length); @@ -860,7 +922,7 @@ static void nvnet_dump_ring_descriptors(NvNetState *s) dma_addr_t rx_ring_addr = nvnet_get_reg(s, NvRegRxRingPhysAddr, 4); rx_ring_addr += i * sizeof(desc); pci_dma_read(d, rx_ring_addr, &desc, sizeof(desc)); - NVNET_DPRINTF("RX: Dumping ring desc %d (%llx): ", + NVNET_DPRINTF("RX: Dumping ring desc %d (%" HWADDR_PRIx "): ", i, rx_ring_addr); NVNET_DPRINTF("Buffer: 0x%x, ", desc.packet_buffer); NVNET_DPRINTF("Length: 0x%x, ", desc.length); diff --git a/hw/xbox/trace-events b/hw/xbox/trace-events index 91e39b3ce28..fbda278fe73 100644 --- a/hw/xbox/trace-events +++ b/hw/xbox/trace-events @@ -7,3 +7,10 @@ nvnet_reg_read(uint32_t addr, const char *name, unsigned int size, uint64_t val) nvnet_reg_write(uint32_t addr, const char *name, unsigned int size, uint64_t val) "addr 0x%"PRIx32" %s size %d val 0x%"PRIx64 nvnet_io_read(uint32_t addr, unsigned int size, uint64_t val) "addr 0x%"PRIx32" size %d val 0x%"PRIx64 nvnet_io_write(uint32_t addr, unsigned int size, uint64_t val) "addr 0x%"PRIx32" size %d val 0x%"PRIx64 +nvnet_rx_filter_bcast_match(void) "broadcast match" +nvnet_rx_filter_mcast_match(uint8_t b0, uint8_t b1, uint8_t b2, uint8_t b3, uint8_t b4, uint8_t b5) "multicast match: %02x:%02x:%02x:%02x:%02x:%02x" +nvnet_rx_filter_mcast_mismatch(uint8_t b0, uint8_t b1, uint8_t b2, uint8_t b3, uint8_t b4, uint8_t b5) "multicast mismatch: %02x:%02x:%02x:%02x:%02x:%02x" +nvnet_rx_filter_ucast_match(uint8_t b0, uint8_t b1, uint8_t b2, uint8_t b3, uint8_t b4, uint8_t b5) "unicast match: %02x:%02x:%02x:%02x:%02x:%02x" +nvnet_rx_filter_ucast_mismatch(uint8_t b0, uint8_t b1, uint8_t b2, uint8_t b3, uint8_t b4, uint8_t b5) "unicast mismatch: %02x:%02x:%02x:%02x:%02x:%02x" +nvnet_rx_oversized(size_t size) "Received packet dropped because it was oversized (%zu bytes)" +nvnet_rx_filter_dropped(void) "Received packet dropped by RX filter" diff --git a/scripts/download-macos-libs.py b/scripts/download-macos-libs.py index 6333e23cc02..239c7ae3e60 100755 --- a/scripts/download-macos-libs.py +++ b/scripts/download-macos-libs.py @@ -19,7 +19,7 @@ class LibInstaller: DARWIN_TARGET_X64="darwin_17" # macOS 10.13 - DARWIN_TARGET_ARM64="darwin_20" # macOS 11.x + DARWIN_TARGET_ARM64="darwin_21" # macOS 12.x def __init__(self, arch): self._queue = [] @@ -44,6 +44,18 @@ def get_latest_pkg_filename_url(self, pkg_name): pkg_base_url = f'{MIRROR}/{pkg_name}' pkg_list = urlopen(pkg_base_url).read().decode('utf-8') pkgs = re.findall(pkg_name + r'[\w\.\-\_\+]*?\.' + self._darwin_target + r'\.' + self._arch + r'\.tbz2', pkg_list) + + if len(pkgs) < 1: + pkgs = re.findall(pkg_name + r'[\w\.\-\_\+]*?\.darwin_any\.' + self._arch + r'\.tbz2', pkg_list) + if len(pkgs) < 1: + pkgs = re.findall(pkg_name + r'[\w\.\-\_\+]*?\.' + self._darwin_target + r'\.noarch\.tbz2', pkg_list) + if len(pkgs) < 1: + pkgs = re.findall(pkg_name + r'[\w\.\-\_\+]*?\.darwin_any\.noarch\.tbz2', pkg_list) + + if len(pkgs) < 1: + print(f' [*] [ERROR] Unable to find version of {pkg_name} compatible with {self._darwin_target}.{self._arch}') + exit(1) + pkg_filename = pkgs[-1] return pkg_filename, f'{pkg_base_url}/{pkg_filename}' diff --git a/softmmu/vl.c b/softmmu/vl.c index cdc485e3d30..badf207da0c 100644 --- a/softmmu/vl.c +++ b/softmmu/vl.c @@ -2853,7 +2853,7 @@ void qemu_init(int argc, char **argv) } if (strlen(dvd_path) > 0) { - if (xemu_check_file(dvd_path)) { + if (xemu_check_file(dvd_path) || strcmp(dvd_path, hdd_path) == 0) { char *msg = g_strdup_printf("Failed to open DVD image file '%s'. Please check machine settings.", dvd_path); xemu_queue_error_message(msg); g_free(msg); diff --git a/ui/meson.build b/ui/meson.build index d09be2dbc02..18bb7c97c18 100644 --- a/ui/meson.build +++ b/ui/meson.build @@ -47,7 +47,7 @@ endif xemu_ss.add(when: 'CONFIG_LINUX', if_true: [gtk, files('xemu-os-utils-linux.c')]) xemu_ss.add(when: 'CONFIG_WIN32', if_true: files('xemu-os-utils-windows.c')) xemu_ss.add(when: 'CONFIG_DARWIN', if_true: files('xemu-os-utils-macos.m')) -xemu_ss.add(imgui, implot, stb_image, noc, sdl, opengl, openssl, fa, fpng, json, httplib) +xemu_ss.add(imgui, implot, stb_image, noc, sdl, opengl, openssl, fa, fpng, json, httplib, fatx) softmmu_ss.add_all(xemu_ss) diff --git a/ui/thirdparty/fatx/fatx.c b/ui/thirdparty/fatx/fatx.c new file mode 100644 index 00000000000..20e934b0f70 --- /dev/null +++ b/ui/thirdparty/fatx/fatx.c @@ -0,0 +1,51 @@ +#include "fatx.h" + +#include "qemu/bswap.h" + +#define FATX_SIGNATURE 0x58544146 + +// This is from libfatx +#pragma pack(1) +struct fatx_superblock { + uint32_t signature; + uint32_t volume_id; + uint32_t sectors_per_cluster; + uint32_t root_cluster; + uint16_t unknown1; + uint8_t padding[4078]; +}; +#pragma pack() + +bool create_fatx_image(const char *filename, unsigned int size) +{ + unsigned int empty_fat = cpu_to_le32(0xfffffff8); + unsigned char zero = 0; + + FILE *fp = qemu_fopen(filename, "wb"); + if (fp != NULL) { + struct fatx_superblock superblock; + memset(&superblock, 0xff, sizeof(struct fatx_superblock)); + + superblock.signature = cpu_to_le32(FATX_SIGNATURE); + superblock.sectors_per_cluster = cpu_to_le32(4); + superblock.volume_id = (uint32_t)rand(); + superblock.root_cluster = cpu_to_le32(1); + superblock.unknown1 = 0; + + // Write the fatx superblock. + fwrite(&superblock, sizeof(superblock), 1, fp); + + // Write the FAT + fwrite(&empty_fat, sizeof(empty_fat), 1, fp); + + fseek(fp, size - sizeof(unsigned char), SEEK_SET); + fwrite(&zero, sizeof(unsigned char), 1, fp); + + fflush(fp); + fclose(fp); + + return true; + } + + return false; +} diff --git a/ui/thirdparty/fatx/fatx.h b/ui/thirdparty/fatx/fatx.h new file mode 100644 index 00000000000..cb4fb053247 --- /dev/null +++ b/ui/thirdparty/fatx/fatx.h @@ -0,0 +1,16 @@ +#ifndef FATX_H +#define FATX_H + +#include "qemu/osdep.h" + +#ifdef __cplusplus +extern "C" { +#endif + +bool create_fatx_image(const char *filename, unsigned int size); + +#ifdef __cplusplus +} +#endif + +#endif diff --git a/ui/thirdparty/meson.build b/ui/thirdparty/meson.build index 6b11d3a9c15..ad01b589322 100644 --- a/ui/thirdparty/meson.build +++ b/ui/thirdparty/meson.build @@ -62,3 +62,6 @@ fpng = declare_dependency(include_directories: 'fpng', link_with: libfpng) json = declare_dependency(include_directories: 'json') httplib = declare_dependency(include_directories: 'httplib') + +libfatx = static_library('fatx', sources: 'fatx/fatx.c') +fatx = declare_dependency(include_directories: 'fatx', link_with: libfatx) diff --git a/ui/thirdparty/noc_file_dialog/noc_file_dialog.h b/ui/thirdparty/noc_file_dialog/noc_file_dialog.h index c41f86b678d..3d8af29c3a9 100644 --- a/ui/thirdparty/noc_file_dialog/noc_file_dialog.h +++ b/ui/thirdparty/noc_file_dialog/noc_file_dialog.h @@ -285,6 +285,9 @@ const char *noc_file_dialog_open(int flags, ofn.lpstrInitialDir = initialDir; ofn.Flags = OFN_PATHMUSTEXIST | OFN_FILEMUSTEXIST | OFN_NOCHANGEDIR; + if (flags & NOC_FILE_DIALOG_OVERWRITE_CONFIRMATION) + ofn.Flags |= OFN_OVERWRITEPROMPT; + if (flags & NOC_FILE_DIALOG_OPEN) { ret = GetOpenFileNameW(&ofn); } else { diff --git a/ui/xemu-input.c b/ui/xemu-input.c index aa932830962..d9181fe2a62 100644 --- a/ui/xemu-input.c +++ b/ui/xemu-input.c @@ -32,6 +32,8 @@ #include "xemu-notifications.h" #include "xemu-settings.h" +#include "sysemu/blockdev.h" + // #define DEBUG_INPUT #ifdef DEBUG_INPUT @@ -93,8 +95,32 @@ static const char **port_index_to_settings_key_map[] = { &g_config.input.bindings.port4, }; +static int *peripheral_types_settings_map[4][2] = { + { &g_config.input.peripherals.port1.peripheral_type_0, + &g_config.input.peripherals.port1.peripheral_type_1 }, + { &g_config.input.peripherals.port2.peripheral_type_0, + &g_config.input.peripherals.port2.peripheral_type_1 }, + { &g_config.input.peripherals.port3.peripheral_type_0, + &g_config.input.peripherals.port3.peripheral_type_1 }, + { &g_config.input.peripherals.port4.peripheral_type_0, + &g_config.input.peripherals.port4.peripheral_type_1 } +}; + +static const char **peripheral_params_settings_map[4][2] = { + { &g_config.input.peripherals.port1.peripheral_param_0, + &g_config.input.peripherals.port1.peripheral_param_1 }, + { &g_config.input.peripherals.port2.peripheral_param_0, + &g_config.input.peripherals.port2.peripheral_param_1 }, + { &g_config.input.peripherals.port3.peripheral_param_0, + &g_config.input.peripherals.port3.peripheral_param_1 }, + { &g_config.input.peripherals.port4.peripheral_param_0, + &g_config.input.peripherals.port4.peripheral_param_1 } +}; + static int sdl_kbd_scancode_map[25]; +static const int port_map[4] = { 3, 4, 1, 2 }; + void xemu_input_init(void) { if (g_config.input.background_input_capture) { @@ -112,6 +138,10 @@ void xemu_input_init(void) new_con->type = INPUT_DEVICE_SDL_KEYBOARD; new_con->name = "Keyboard"; new_con->bound = -1; + new_con->peripheral_types[0] = PERIPHERAL_NONE; + new_con->peripheral_types[1] = PERIPHERAL_NONE; + new_con->peripherals[0] = NULL; + new_con->peripherals[1] = NULL; sdl_kbd_scancode_map[0] = g_config.input.keyboard_controller_scancode_map.a; sdl_kbd_scancode_map[1] = g_config.input.keyboard_controller_scancode_map.b; @@ -154,6 +184,7 @@ void xemu_input_init(void) char buf[128]; snprintf(buf, sizeof(buf), "Connected '%s' to port %d", new_con->name, port+1); xemu_queue_notification(buf); + xemu_input_rebind_xmu(port); } QTAILQ_INSERT_TAIL(&available_controllers, new_con, entry); @@ -177,6 +208,24 @@ int xemu_input_get_controller_default_bind_port(ControllerState *state, int star return -1; } +void xemu_save_peripheral_settings(int player_index, int peripheral_index, + int peripheral_type, + const char *peripheral_parameter) +{ + int *peripheral_type_ptr = + peripheral_types_settings_map[player_index][peripheral_index]; + const char **peripheral_param_ptr = + peripheral_params_settings_map[player_index][peripheral_index]; + + assert(peripheral_type_ptr); + assert(peripheral_param_ptr); + + *peripheral_type_ptr = peripheral_type; + xemu_settings_set_string( + peripheral_param_ptr, + peripheral_parameter == NULL ? "" : peripheral_parameter); +} + void xemu_input_process_sdl_events(const SDL_Event *event) { if (event->type == SDL_CONTROLLERDEVICEADDED) { @@ -201,6 +250,10 @@ void xemu_input_process_sdl_events(const SDL_Event *event) new_con->sdl_joystick_id = SDL_JoystickInstanceID(new_con->sdl_joystick); new_con->sdl_joystick_guid = SDL_JoystickGetGUID(new_con->sdl_joystick); new_con->bound = -1; + new_con->peripheral_types[0] = PERIPHERAL_NONE; + new_con->peripheral_types[1] = PERIPHERAL_NONE; + new_con->peripherals[0] = NULL; + new_con->peripherals[1] = NULL; char guid_buf[35] = { 0 }; SDL_JoystickGetGUIDString(new_con->sdl_joystick_guid, guid_buf, sizeof(guid_buf)); @@ -254,6 +307,7 @@ void xemu_input_process_sdl_events(const SDL_Event *event) char buf[128]; snprintf(buf, sizeof(buf), "Connected '%s' to port %d", new_con->name, port+1); xemu_queue_notification(buf); + xemu_input_rebind_xmu(port); } } else if (event->type == SDL_CONTROLLERDEVICEREMOVED) { DPRINTF("Controller Removed: %d\n", event->cdevice.which); @@ -286,6 +340,11 @@ void xemu_input_process_sdl_events(const SDL_Event *event) if (iter->sdl_gamecontroller) { SDL_GameControllerClose(iter->sdl_gamecontroller); } + + for (int i = 0; i < 2; i++) { + if (iter->peripherals[i]) + g_free(iter->peripherals[i]); + } free(iter); handled = 1; @@ -380,12 +439,9 @@ void xemu_input_update_sdl_controller_state(ControllerState *state) } const SDL_GameControllerAxis sdl_axis_map[6] = { - SDL_CONTROLLER_AXIS_TRIGGERLEFT, - SDL_CONTROLLER_AXIS_TRIGGERRIGHT, - SDL_CONTROLLER_AXIS_LEFTX, - SDL_CONTROLLER_AXIS_LEFTY, - SDL_CONTROLLER_AXIS_RIGHTX, - SDL_CONTROLLER_AXIS_RIGHTY, + SDL_CONTROLLER_AXIS_TRIGGERLEFT, SDL_CONTROLLER_AXIS_TRIGGERRIGHT, + SDL_CONTROLLER_AXIS_LEFTX, SDL_CONTROLLER_AXIS_LEFTY, + SDL_CONTROLLER_AXIS_RIGHTX, SDL_CONTROLLER_AXIS_RIGHTY, }; for (int i = 0; i < 6; i++) { @@ -429,6 +485,22 @@ void xemu_input_bind(int index, ControllerState *state, int save) if (bound_controllers[index]) { assert(bound_controllers[index]->device != NULL); Error *err = NULL; + + // Unbind any XMUs + for (int i = 0; i < 2; i++) { + if (bound_controllers[index]->peripherals[i]) { + // If this was an XMU, unbind the XMU + if (bound_controllers[index]->peripheral_types[i] == + PERIPHERAL_XMU) + xemu_input_unbind_xmu(index, i); + + // Free up the XmuState and set the peripheral type to none + g_free(bound_controllers[index]->peripherals[i]); + bound_controllers[index]->peripherals[i] = NULL; + bound_controllers[index]->peripheral_types[i] = PERIPHERAL_NONE; + } + } + qdev_unplug((DeviceState *)bound_controllers[index]->device, &err); assert(err == NULL); @@ -460,7 +532,6 @@ void xemu_input_bind(int index, ControllerState *state, int save) bound_controllers[index] = state; bound_controllers[index]->bound = index; - const int port_map[4] = {3, 4, 1, 2}; char *tmp; // Create controller's internal USB hub. @@ -506,6 +577,179 @@ void xemu_input_bind(int index, ControllerState *state, int save) } } +bool xemu_input_bind_xmu(int player_index, int expansion_slot_index, + const char *filename, bool is_rebind) +{ + assert(player_index >= 0 && player_index < 4); + assert(expansion_slot_index >= 0 && expansion_slot_index < 2); + + ControllerState *player = bound_controllers[player_index]; + enum peripheral_type peripheral_type = + player->peripheral_types[expansion_slot_index]; + if (peripheral_type != PERIPHERAL_XMU) + return false; + + XmuState *xmu = (XmuState *)player->peripherals[expansion_slot_index]; + + // Unbind existing XMU + if (xmu->dev != NULL) { + xemu_input_unbind_xmu(player_index, expansion_slot_index); + } + + if (filename == NULL) + return false; + + // Look for any other XMUs that are using this file, and unbind them + for (int player_i = 0; player_i < 4; player_i++) { + ControllerState *state = bound_controllers[player_i]; + if (state != NULL) { + for (int peripheral_i = 0; peripheral_i < 2; peripheral_i++) { + if (state->peripheral_types[peripheral_i] == PERIPHERAL_XMU) { + XmuState *xmu_i = + (XmuState *)state->peripherals[peripheral_i]; + assert(xmu_i); + + if (xmu_i->filename != NULL && + strcmp(xmu_i->filename, filename) == 0) { + char *buf = + g_strdup_printf("This XMU is already mounted on " + "player %d slot %c\r\n", + player_i + 1, 'A' + peripheral_i); + xemu_queue_notification(buf); + g_free(buf); + return false; + } + } + } + } + } + + xmu->filename = g_strdup(filename); + + const int xmu_map[2] = { 2, 3 }; + char *tmp; + + static int id_counter = 0; + tmp = g_strdup_printf("xmu_%d", id_counter++); + + // Add the file as a drive + QDict *qdict1 = qdict_new(); + qdict_put_str(qdict1, "id", tmp); + qdict_put_str(qdict1, "format", "raw"); + qdict_put_str(qdict1, "file", filename); + + QemuOpts *drvopts = + qemu_opts_from_qdict(qemu_find_opts("drive"), qdict1, &error_abort); + + DriveInfo *dinfo = drive_new(drvopts, 0, &error_abort); + assert(dinfo); + + // Create the usb-storage device + QDict *qdict2 = qdict_new(); + + // Specify device driver + qdict_put_str(qdict2, "driver", "usb-storage"); + + // Specify device identifier + qdict_put_str(qdict2, "drive", tmp); + g_free(tmp); + + // Specify index/port + tmp = g_strdup_printf("1.%d.%d", port_map[player_index], + xmu_map[expansion_slot_index]); + qdict_put_str(qdict2, "port", tmp); + g_free(tmp); + + // Create the device + QemuOpts *opts = + qemu_opts_from_qdict(qemu_find_opts("device"), qdict2, &error_abort); + + DeviceState *dev = qdev_device_add(opts, &error_abort); + assert(dev); + + xmu->dev = (void *)dev; + + // Unref for eventual cleanup + qobject_unref(qdict1); + qobject_unref(qdict2); + + if (!is_rebind) { + xemu_save_peripheral_settings(player_index, expansion_slot_index, + peripheral_type, xmu->filename); + } + + return true; +} + +void xemu_input_unbind_xmu(int player_index, int expansion_slot_index) +{ + assert(player_index >= 0 && player_index < 4); + assert(expansion_slot_index >= 0 && expansion_slot_index < 2); + + ControllerState *state = bound_controllers[player_index]; + if (state->peripheral_types[expansion_slot_index] != PERIPHERAL_XMU) + return; + + XmuState *xmu = (XmuState *)state->peripherals[expansion_slot_index]; + if (xmu != NULL) { + if (xmu->dev != NULL) { + qdev_unplug((DeviceState *)xmu->dev, &error_abort); + object_unref(OBJECT(xmu->dev)); + xmu->dev = NULL; + } + + g_free((void *)xmu->filename); + xmu->filename = NULL; + } +} + +void xemu_input_rebind_xmu(int port) +{ + // Try to bind peripherals back to controller + for (int i = 0; i < 2; i++) { + enum peripheral_type peripheral_type = + (enum peripheral_type)(*peripheral_types_settings_map[port][i]); + + // If peripheralType is out of range, change the settings for this + // controller and peripheral port to default + if (peripheral_type < PERIPHERAL_NONE || + peripheral_type >= PERIPHERAL_TYPE_COUNT) { + xemu_save_peripheral_settings(port, i, PERIPHERAL_NONE, NULL); + peripheral_type = PERIPHERAL_NONE; + } + + const char *param = *peripheral_params_settings_map[port][i]; + + if (peripheral_type == PERIPHERAL_XMU) { + if (param != NULL && strlen(param) > 0) { + // This is an XMU and needs to be bound to this controller + if (qemu_access(param, R_OK | W_OK) == 0) { + bound_controllers[port]->peripheral_types[i] = + peripheral_type; + bound_controllers[port]->peripherals[i] = + g_malloc(sizeof(XmuState)); + memset(bound_controllers[port]->peripherals[i], 0, + sizeof(XmuState)); + bool did_bind = xemu_input_bind_xmu(port, i, param, true); + if (did_bind) { + char *buf = + g_strdup_printf("Connected XMU %s to port %d%c", + param, port + 1, 'A' + i); + xemu_queue_notification(buf); + g_free(buf); + } + } else { + char *buf = + g_strdup_printf("Unable to bind XMU at %s to port %d%c", + param, port + 1, 'A' + i); + xemu_queue_error_message(buf); + g_free(buf); + } + } + } + } +} + void xemu_input_set_test_mode(int enabled) { test_mode = enabled; diff --git a/ui/xemu-input.h b/ui/xemu-input.h index 8a8ba6544ea..330ae58a7c1 100644 --- a/ui/xemu-input.h +++ b/ui/xemu-input.h @@ -26,6 +26,8 @@ #define XEMU_INPUT_H #include +#include + #include "qemu/queue.h" enum controller_state_buttons_mask { @@ -63,6 +65,13 @@ enum controller_input_device_type { INPUT_DEVICE_SDL_GAMECONTROLLER, }; +enum peripheral_type { PERIPHERAL_NONE, PERIPHERAL_XMU, PERIPHERAL_TYPE_COUNT }; + +typedef struct XmuState { + const char *filename; + void *dev; +} XmuState; + typedef struct ControllerState { QTAILQ_ENTRY(ControllerState) entry; @@ -88,6 +97,9 @@ typedef struct ControllerState { SDL_JoystickID sdl_joystick_id; SDL_JoystickGUID sdl_joystick_guid; + enum peripheral_type peripheral_types[2]; + void *peripherals[2]; + int bound; // Which port this input device is bound to void *device; // DeviceState opaque } ControllerState; @@ -109,7 +121,14 @@ void xemu_input_update_sdl_controller_state(ControllerState *state); void xemu_input_update_rumble(ControllerState *state); ControllerState *xemu_input_get_bound(int index); void xemu_input_bind(int index, ControllerState *state, int save); +bool xemu_input_bind_xmu(int player_index, int peripheral_port_index, + const char *filename, bool is_rebind); +void xemu_input_rebind_xmu(int port); +void xemu_input_unbind_xmu(int player_index, int peripheral_port_index); int xemu_input_get_controller_default_bind_port(ControllerState *state, int start); +void xemu_save_peripheral_settings(int player_index, int peripheral_index, + int peripheral_type, + const char *peripheral_parameter); void xemu_input_set_test_mode(int enabled); int xemu_input_get_test_mode(void); diff --git a/ui/xui/debug.cc b/ui/xui/debug.cc index f44c1af790f..2d11eea2ed1 100644 --- a/ui/xui/debug.cc +++ b/ui/xui/debug.cc @@ -230,6 +230,9 @@ DebugVideoWindow::DebugVideoWindow() { m_is_open = false; m_transparent = false; + m_position_restored = false; + m_resize_init_complete = false; + m_prev_scale = g_viewport_mgr.m_scale; } void DebugVideoWindow::Draw() @@ -237,9 +240,25 @@ void DebugVideoWindow::Draw() if (!m_is_open) return; + if (!m_position_restored) { + ImGui::SetNextWindowPos(ImVec2(g_config.display.debug.video.x_pos, + g_config.display.debug.video.y_pos), + ImGuiCond_Once, ImVec2(0, 0)); + m_transparent = g_config.display.debug.video.transparency; + m_position_restored = true; + } + float alpha = m_transparent ? 0.2 : 1.0; PushWindowTransparencySettings(m_transparent, 0.2); - ImGui::SetNextWindowSize(ImVec2(600.0f*g_viewport_mgr.m_scale, 150.0f*g_viewport_mgr.m_scale), ImGuiCond_Once); + + if (!m_resize_init_complete || (g_viewport_mgr.m_scale != m_prev_scale)) { + ImGui::SetNextWindowSize(ImVec2( + g_config.display.debug.video.x_winsize * g_viewport_mgr.m_scale, + g_config.display.debug.video.y_winsize * g_viewport_mgr.m_scale)); + m_resize_init_complete = true; + } + m_prev_scale = g_viewport_mgr.m_scale; + if (ImGui::Begin("Video Debug", &m_is_open)) { double x_start, x_end; static ImPlotAxisFlags rt_axis = ImPlotAxisFlags_NoTickLabels; @@ -287,7 +306,12 @@ void DebugVideoWindow::Draw() } ImPlot::PopStyleColor(); - if (ImGui::TreeNode("Advanced")) { + ImGui::SetNextItemOpen(g_config.display.debug.video.advanced_tree_state, + ImGuiCond_Once); + g_config.display.debug.video.advanced_tree_state = + ImGui::TreeNode("Advanced"); + + if (g_config.display.debug.video.advanced_tree_state) { ImGui::SetNextWindowBgAlpha(alpha); if (ImPlot::BeginPlot("##ScrollingDraws", ImVec2(-1,-1))) { ImPlot::SetupAxes(NULL, NULL, ImPlotAxisFlags_None, ImPlotAxisFlags_AutoFit); @@ -326,6 +350,17 @@ void DebugVideoWindow::Draw() } ImPlot::PopStyleVar(2); + + ImVec2 debug_window_pos = ImGui::GetWindowPos(); + g_config.display.debug.video.x_pos = debug_window_pos.x; + g_config.display.debug.video.y_pos = debug_window_pos.y; + + ImVec2 debug_window_size = ImGui::GetWindowSize(); + g_config.display.debug.video.x_winsize = + debug_window_size.x / g_viewport_mgr.m_scale; + g_config.display.debug.video.y_winsize = + debug_window_size.y / g_viewport_mgr.m_scale; + g_config.display.debug.video.transparency = m_transparent; } ImGui::End(); ImGui::PopStyleColor(5); diff --git a/ui/xui/debug.hh b/ui/xui/debug.hh index 92671dceffd..056deaae6ff 100644 --- a/ui/xui/debug.hh +++ b/ui/xui/debug.hh @@ -31,6 +31,9 @@ class DebugVideoWindow public: bool m_is_open; bool m_transparent; + bool m_position_restored; + bool m_resize_init_complete; + float m_prev_scale; DebugVideoWindow(); void Draw(); diff --git a/ui/xui/gl-helpers.cc b/ui/xui/gl-helpers.cc index 4155ff20979..6c07d087b41 100644 --- a/ui/xui/gl-helpers.cc +++ b/ui/xui/gl-helpers.cc @@ -16,25 +16,24 @@ // You should have received a copy of the GNU General Public License // along with this program. If not, see . // -#include "common.hh" -#include -#include -#include -#include +#include "ui/xemu-widescreen.h" #include "gl-helpers.hh" -#include "stb_image.h" +#include "common.hh" #include "data/controller_mask.png.h" #include "data/logo_sdf.png.h" -#include "ui/shader/xemu-logo-frag.h" #include "data/xemu_64x64.png.h" +#include "data/xmu_mask.png.h" #include "notifications.hh" -#include "ui/xemu-widescreen.h" +#include "stb_image.h" +#include +#include +#include +#include -Fbo *controller_fbo, - *logo_fbo; -GLuint g_controller_tex, - g_logo_tex, - g_icon_tex; +#include "ui/shader/xemu-logo-frag.h" + +Fbo *controller_fbo, *xmu_fbo, *logo_fbo; +GLuint g_controller_tex, g_logo_tex, g_icon_tex, g_xmu_tex; enum class ShaderType { Blit, @@ -186,9 +185,11 @@ static GLuint Shader(GLenum type, const char *src) glGetShaderiv(shader, GL_COMPILE_STATUS, &status); if (status != GL_TRUE) { glGetShaderInfoLog(shader, sizeof(err_buf), NULL, err_buf); - fprintf(stderr, "Shader compilation failed: %s\n\n" - "[Shader Source]\n" - "%s\n", err_buf, src); + fprintf(stderr, + "Shader compilation failed: %s\n\n" + "[Shader Source]\n" + "%s\n", + err_buf, src); assert(0); } @@ -222,15 +223,15 @@ void main() { GLuint vert = Shader(GL_VERTEX_SHADER, vert_src); assert(vert != 0); -// const char *image_frag_src = R"( -// #version 150 core -// uniform sampler2D tex; -// in vec2 Texcoord; -// out vec4 out_Color; -// void main() { -// out_Color.rgba = texture(tex, Texcoord); -// } -// )"; + // const char *image_frag_src = R"( + // #version 150 core + // uniform sampler2D tex; + // in vec2 Texcoord; + // out vec4 out_Color; + // void main() { + // out_Color.rgba = texture(tex, Texcoord); + // } + // )"; const char *image_gamma_frag_src = R"( #version 400 core @@ -370,7 +371,7 @@ void RenderDecal(DecalShader *s, float x, float y, float w, float h, glGetTexLevelParameteriv(GL_TEXTURE_2D, 0, GL_TEXTURE_HEIGHT, &th_i); float tw = tw_i, th = th_i; - #define COL(color, c) (float)(((color)>>((c)*8)) & 0xff)/255.0 +#define COL(color, c) (float)(((color) >> ((c)*8)) & 0xff) / 255.0 if (s->flipy_loc >= 0) { glUniform1i(s->flipy_loc, s->flip); } @@ -403,7 +404,7 @@ void RenderDecal(DecalShader *s, float x, float y, float w, float h, if (s->scale_loc >= 0) { glUniform1f(s->scale_loc, s->scale); } - #undef COL +#undef COL glDrawElements(GL_TRIANGLE_FAN, 4, GL_UNSIGNED_INT, NULL); } @@ -412,14 +413,15 @@ struct rect { }; static const struct rect tex_items[] = { - { 0, 148, 467, 364 }, // obj_controller - { 0, 81, 67, 67 }, // obj_lstick - { 0, 14, 67, 67 }, // obj_rstick - { 67, 104, 68, 44 }, // obj_port_socket - { 67, 76, 28, 28 }, // obj_port_lbl_1 - { 67, 48, 28, 28 }, // obj_port_lbl_2 - { 67, 20, 28, 28 }, // obj_port_lbl_3 - { 95, 76, 28, 28 }, // obj_port_lbl_4 + { 0, 148, 467, 364 }, // obj_controller + { 0, 81, 67, 67 }, // obj_lstick + { 0, 14, 67, 67 }, // obj_rstick + { 67, 104, 68, 44 }, // obj_port_socket + { 67, 76, 28, 28 }, // obj_port_lbl_1 + { 67, 48, 28, 28 }, // obj_port_lbl_2 + { 67, 20, 28, 28 }, // obj_port_lbl_3 + { 95, 76, 28, 28 }, // obj_port_lbl_4 + { 0, 0, 512, 512 } // obj_xmu }; enum tex_item_names { @@ -431,6 +433,7 @@ enum tex_item_names { obj_port_lbl_2, obj_port_lbl_3, obj_port_lbl_4, + obj_xmu }; void InitCustomRendering(void) @@ -441,6 +444,9 @@ void InitCustomRendering(void) g_decal_shader = NewDecalShader(ShaderType::Mask); controller_fbo = new Fbo(512, 512); + g_xmu_tex = LoadTextureFromMemory(xmu_mask_data, xmu_mask_size); + xmu_fbo = new Fbo(512, 256); + g_logo_tex = LoadTextureFromMemory(logo_sdf_data, logo_sdf_size); g_logo_shader = NewDecalShader(ShaderType::Logo); logo_fbo = new Fbo(512, 512); @@ -646,6 +652,26 @@ void RenderControllerPort(float frame_x, float frame_y, int i, glUseProgram(0); } +void RenderXmu(float frame_x, float frame_y, uint32_t primary_color, + uint32_t secondary_color) +{ + glUseProgram(g_decal_shader->prog); + glBindVertexArray(g_decal_shader->vao); + glActiveTexture(GL_TEXTURE0); + glBindTexture(GL_TEXTURE_2D, g_xmu_tex); + glBlendEquation(GL_FUNC_ADD); + glBlendFunc(GL_ONE, GL_ZERO); + + // Render xmu + RenderDecal(g_decal_shader, frame_x, frame_y, 256, 256, + tex_items[obj_xmu].x, tex_items[obj_xmu].y, + tex_items[obj_xmu].w, tex_items[obj_xmu].h, primary_color, + secondary_color, 0); + + glBindVertexArray(0); + glUseProgram(0); +} + void RenderLogo(uint32_t time) { uint32_t color = 0x62ca13ff; @@ -801,8 +827,7 @@ void SaveScreenshot(GLuint tex, bool flip) time_t t = time(NULL); struct tm *tmp = localtime(&t); if (tmp) { - strftime(fname, sizeof(fname), "xemu-%Y-%m-%d-%H-%M-%S.png", - tmp); + strftime(fname, sizeof(fname), "xemu-%Y-%m-%d-%H-%M-%S.png", tmp); } else { strcpy(fname, "xemu.png"); } diff --git a/ui/xui/gl-helpers.hh b/ui/xui/gl-helpers.hh index 74015408166..82da963e6ba 100644 --- a/ui/xui/gl-helpers.hh +++ b/ui/xui/gl-helpers.hh @@ -38,7 +38,7 @@ public: void Restore(); }; -extern Fbo *controller_fbo, *logo_fbo; +extern Fbo *controller_fbo, *xmu_fbo, *logo_fbo; extern GLuint g_icon_tex; void InitCustomRendering(void); @@ -47,6 +47,8 @@ void RenderController(float frame_x, float frame_y, uint32_t primary_color, uint32_t secondary_color, ControllerState *state); void RenderControllerPort(float frame_x, float frame_y, int i, uint32_t port_color); +void RenderXmu(float frame_x, float frame_y, uint32_t primary_color, + uint32_t secondary_color); void RenderFramebuffer(GLint tex, int width, int height, bool flip); void RenderFramebuffer(GLint tex, int width, int height, bool flip, float scale[2]); bool RenderFramebufferToPng(GLuint tex, bool flip, std::vector &png, int max_width = 0, int max_height = 0); diff --git a/ui/xui/main-menu.cc b/ui/xui/main-menu.cc index f29b0139832..75b88cafb6e 100644 --- a/ui/xui/main-menu.cc +++ b/ui/xui/main-menu.cc @@ -40,6 +40,10 @@ #include "../xemu-os-utils.h" #include "../xemu-xbe.h" +#include "../thirdparty/fatx/fatx.h" + +#define DEFAULT_XMU_SIZE 8388608 + MainMenuScene g_main_menu; MainMenuTabView::~MainMenuTabView() {} @@ -86,6 +90,9 @@ void MainMenuInputView::Draw() // Dimensions of controller (rendered at origin) float controller_width = 477.0f; float controller_height = 395.0f; + // Dimensions of XMU + float xmu_x = 0, xmu_x_stride = 256, xmu_y = 0; + float xmu_w = 256, xmu_h = 256; // Setup rendering to fbo for controller and port images controller_fbo->Target(); @@ -120,14 +127,14 @@ void MainMenuInputView::Draw() // uses the texture as a unique ID. Push a new ID now to resolve // the conflict. ImGui::PushID(i); - float x = b_x+i*b_x_stride; - ImGui::PushStyleColor(ImGuiCol_Button, is_selected ? - color_active : - color_inactive); - bool activated = ImGui::ImageButton(id, - ImVec2(b_w*g_viewport_mgr.m_scale,b_h*g_viewport_mgr.m_scale), - ImVec2(x/t_w, (b_y+b_h)/t_h), - ImVec2((x+b_w)/t_w, b_y/t_h), + float x = b_x + i * b_x_stride; + ImGui::PushStyleColor(ImGuiCol_Button, + is_selected ? color_active : color_inactive); + bool activated = ImGui::ImageButton( + id, + ImVec2(b_w * g_viewport_mgr.m_scale, b_h * g_viewport_mgr.m_scale), + ImVec2(x / t_w, (b_y + b_h) / t_h), + ImVec2((x + b_w) / t_w, b_y / t_h), port_padding * g_viewport_mgr.m_scale); ImGui::PopStyleColor(); @@ -193,6 +200,16 @@ void MainMenuInputView::Draw() } if (ImGui::Selectable(selectable_label, is_selected)) { xemu_input_bind(active, iter, 1); + + // FIXME: We want to bind the XMU here, but we can't because we + // just unbound it and we need to wait for Qemu to release the + // file + + // If we previously had no controller connected, we can rebind + // the XMU + if (bound_state == NULL) + xemu_input_rebind_xmu(active); + bound_state = iter; } if (is_selected) { @@ -260,6 +277,168 @@ void MainMenuInputView::Draw() ImGui::PopFont(); ImGui::SetCursorPos(pos); + if (bound_state) { + SectionTitle("Expansion Slots"); + // Begin a 2-column layout to render the expansion slots + ImGui::PushStyleVar(ImGuiStyleVar_ItemSpacing, + g_viewport_mgr.Scale(ImVec2(0, 12))); + ImGui::Columns(2, "mixed", false); + + xmu_fbo->Target(); + id = (ImTextureID)(intptr_t)xmu_fbo->Texture(); + + const char *img_file_filters = ".img Files\0*.img\0All Files\0*.*\0"; + const char *comboLabels[2] = { "###ExpansionSlotA", + "###ExpansionSlotB" }; + for (int i = 0; i < 2; i++) { + // Display a combo box to allow the user to choose the type of + // peripheral they want to use + enum peripheral_type selected_type = + bound_state->peripheral_types[i]; + const char *peripheral_type_names[2] = { "None", "Memory Unit" }; + const char *selected_peripheral_type = + peripheral_type_names[selected_type]; + ImGui::SetNextItemWidth(-FLT_MIN); + if (ImGui::BeginCombo(comboLabels[i], selected_peripheral_type, + ImGuiComboFlags_NoArrowButton)) { + // Handle all available peripheral types + for (int j = 0; j < 2; j++) { + bool is_selected = selected_type == j; + ImGui::PushID(j); + const char *selectable_label = peripheral_type_names[j]; + + if (ImGui::Selectable(selectable_label, is_selected)) { + // Free any existing peripheral + if (bound_state->peripherals[i] != NULL) { + if (bound_state->peripheral_types[i] == + PERIPHERAL_XMU) { + // Another peripheral was already bound. + // Unplugging + xemu_input_unbind_xmu(active, i); + } + + // Free the existing state + g_free((void *)bound_state->peripherals[i]); + bound_state->peripherals[i] = NULL; + } + + // Change the peripheral type to the newly selected type + bound_state->peripheral_types[i] = + (enum peripheral_type)j; + + // Allocate state for the new peripheral + if (j == PERIPHERAL_XMU) { + bound_state->peripherals[i] = + g_malloc(sizeof(XmuState)); + memset(bound_state->peripherals[i], 0, + sizeof(XmuState)); + } + + xemu_save_peripheral_settings( + active, i, bound_state->peripheral_types[i], NULL); + } + + if (is_selected) { + ImGui::SetItemDefaultFocus(); + } + + ImGui::PopID(); + } + + ImGui::EndCombo(); + } + DrawComboChevron(); + + // Set an X offset to center the image button within the column + ImGui::SetCursorPosX( + ImGui::GetCursorPosX() + + (int)((ImGui::GetColumnWidth() - + xmu_w * g_viewport_mgr.m_scale - + 2 * port_padding * g_viewport_mgr.m_scale) / + 2)); + + selected_type = bound_state->peripheral_types[i]; + if (selected_type == PERIPHERAL_XMU) { + float x = xmu_x + i * xmu_x_stride; + float y = xmu_y; + + XmuState *xmu = (XmuState *)bound_state->peripherals[i]; + if (xmu->filename != NULL && strlen(xmu->filename) > 0) { + RenderXmu(x, y, 0x81dc8a00, 0x0f0f0f00); + + } else { + RenderXmu(x, y, 0x1f1f1f00, 0x0f0f0f00); + } + + ImVec2 xmu_display_size; + if (ImGui::GetContentRegionMax().x < + xmu_h * g_viewport_mgr.m_scale) { + xmu_display_size.x = ImGui::GetContentRegionMax().x / 2; + xmu_display_size.y = xmu_display_size.x * xmu_h / xmu_w; + } else { + xmu_display_size = ImVec2(xmu_w * g_viewport_mgr.m_scale, + xmu_h * g_viewport_mgr.m_scale); + } + + ImGui::SetCursorPosX( + ImGui::GetCursorPosX() + + (int)((ImGui::GetColumnWidth() - xmu_display_size.x) / + 2.0)); + + ImGui::Image(id, xmu_display_size, ImVec2(0.5f * i, 1), + ImVec2(0.5f * (i + 1), 0)); + ImVec2 pos = ImGui::GetCursorPos(); + + ImGui::SetCursorPos(pos); + + // Button to generate a new XMU + ImGui::PushID(i); + if (ImGui::Button("New Image", ImVec2(250, 0))) { + int flags = NOC_FILE_DIALOG_SAVE | + NOC_FILE_DIALOG_OVERWRITE_CONFIRMATION; + const char *new_path = PausedFileOpen( + flags, img_file_filters, NULL, "xmu.img"); + + if (new_path) { + if (create_fatx_image(new_path, DEFAULT_XMU_SIZE)) { + // XMU was created successfully. Bind it + xemu_input_bind_xmu(active, i, new_path, false); + } else { + // Show alert message + char *msg = g_strdup_printf( + "Unable to create XMU image at %s", new_path); + xemu_queue_error_message(msg); + g_free(msg); + } + } + } + + const char *xmu_port_path = NULL; + if (xmu->filename == NULL) + xmu_port_path = g_strdup(""); + else + xmu_port_path = g_strdup(xmu->filename); + if (FilePicker("Image", &xmu_port_path, img_file_filters)) { + if (strlen(xmu_port_path) == 0) { + xemu_input_unbind_xmu(active, i); + } else { + xemu_input_bind_xmu(active, i, xmu_port_path, false); + } + } + g_free((void *)xmu_port_path); + + ImGui::PopID(); + } + + ImGui::NextColumn(); + } + + xmu_fbo->Restore(); + + ImGui::PopStyleVar(); // ItemSpacing + ImGui::Columns(1); + } + SectionTitle("Options"); Toggle("Auto-bind controllers", &g_config.input.auto_bind, "Bind newly connected controllers to any open port"); @@ -625,8 +804,9 @@ void MainMenuNetworkView::DrawNatOptions(bool appearing) void MainMenuNetworkView::DrawUdpOptions(bool appearing) { if (appearing) { - strncpy(remote_addr, g_config.net.udp.remote_addr, sizeof(remote_addr)-1); - strncpy(local_addr, g_config.net.udp.bind_addr, sizeof(local_addr)-1); + strncpy(remote_addr, g_config.net.udp.remote_addr, + sizeof(remote_addr) - 1); + strncpy(local_addr, g_config.net.udp.bind_addr, sizeof(local_addr) - 1); } float size_ratio = 0.5; @@ -663,7 +843,9 @@ MainMenuSnapshotsView::~MainMenuSnapshotsView() g_free(m_search_regex); } -bool MainMenuSnapshotsView::BigSnapshotButton(QEMUSnapshotInfo *snapshot, XemuSnapshotData *data, int current_snapshot_binding) +bool MainMenuSnapshotsView::BigSnapshotButton(QEMUSnapshotInfo *snapshot, + XemuSnapshotData *data, + int current_snapshot_binding) { ImGuiStyle &style = ImGui::GetStyle(); ImDrawList *draw_list = ImGui::GetWindowDrawList(); @@ -673,18 +855,27 @@ bool MainMenuSnapshotsView::BigSnapshotButton(QEMUSnapshotInfo *snapshot, XemuSn ImGui::PopFont(); ImGui::PushStyleVar(ImGuiStyleVar_ButtonTextAlign, ImVec2(0, 0)); - ImGui::PushStyleVar(ImGuiStyleVar_FramePadding, g_viewport_mgr.Scale(ImVec2(5, 5))); + ImGui::PushStyleVar(ImGuiStyleVar_FramePadding, + g_viewport_mgr.Scale(ImVec2(5, 5))); ImGui::PushFont(g_font_mgr.m_menu_font_medium); ImVec2 ts_title = ImGui::CalcTextSize(snapshot->name); - ImVec2 thumbnail_size = g_viewport_mgr.Scale(ImVec2(XEMU_SNAPSHOT_THUMBNAIL_WIDTH, XEMU_SNAPSHOT_THUMBNAIL_HEIGHT)); + ImVec2 thumbnail_size = g_viewport_mgr.Scale( + ImVec2(XEMU_SNAPSHOT_THUMBNAIL_WIDTH, XEMU_SNAPSHOT_THUMBNAIL_HEIGHT)); ImVec2 thumbnail_pos(style.FramePadding.x, style.FramePadding.y); - ImVec2 name_pos(thumbnail_pos.x + thumbnail_size.x + style.FramePadding.x * 2, thumbnail_pos.y); - ImVec2 title_pos(name_pos.x, name_pos.y + ts_title.y + style.FramePadding.x); - ImVec2 date_pos(name_pos.x, title_pos.y + ts_title.y + style.FramePadding.x); - ImVec2 binding_pos(name_pos.x, date_pos.y + ts_title.y + style.FramePadding.x); - ImVec2 button_size(-FLT_MIN, fmax(thumbnail_size.y + style.FramePadding.y * 2, ts_title.y + ts_sub.y + style.FramePadding.y * 3)); + ImVec2 name_pos(thumbnail_pos.x + thumbnail_size.x + + style.FramePadding.x * 2, + thumbnail_pos.y); + ImVec2 title_pos(name_pos.x, + name_pos.y + ts_title.y + style.FramePadding.x); + ImVec2 date_pos(name_pos.x, + title_pos.y + ts_title.y + style.FramePadding.x); + ImVec2 binding_pos(name_pos.x, + date_pos.y + ts_title.y + style.FramePadding.x); + ImVec2 button_size(-FLT_MIN, + fmax(thumbnail_size.y + style.FramePadding.y * 2, + ts_title.y + ts_sub.y + style.FramePadding.y * 3)); bool load = ImGui::Button("###button", button_size); @@ -699,42 +890,55 @@ bool MainMenuSnapshotsView::BigSnapshotButton(QEMUSnapshotInfo *snapshot, XemuSn int thumbnail_width, thumbnail_height; glActiveTexture(GL_TEXTURE0); glBindTexture(GL_TEXTURE_2D, thumbnail); - glGetTexLevelParameteriv(GL_TEXTURE_2D, 0, GL_TEXTURE_WIDTH, &thumbnail_width); - glGetTexLevelParameteriv(GL_TEXTURE_2D, 0, GL_TEXTURE_HEIGHT, &thumbnail_height); + glGetTexLevelParameteriv(GL_TEXTURE_2D, 0, GL_TEXTURE_WIDTH, + &thumbnail_width); + glGetTexLevelParameteriv(GL_TEXTURE_2D, 0, GL_TEXTURE_HEIGHT, + &thumbnail_height); // Draw black background behind thumbnail ImVec2 thumbnail_min(p0.x + thumbnail_pos.x, p0.y + thumbnail_pos.y); - ImVec2 thumbnail_max(thumbnail_min.x + thumbnail_size.x, thumbnail_min.y + thumbnail_size.y); + ImVec2 thumbnail_max(thumbnail_min.x + thumbnail_size.x, + thumbnail_min.y + thumbnail_size.y); draw_list->AddRectFilled(thumbnail_min, thumbnail_max, IM_COL32_BLACK); // Draw centered thumbnail image int scaled_width, scaled_height; - ScaleDimensions(thumbnail_width, thumbnail_height, thumbnail_size.x, thumbnail_size.y, &scaled_width, &scaled_height); - ImVec2 img_min = ImVec2(thumbnail_min.x + (thumbnail_size.x - scaled_width) / 2, - thumbnail_min.y + (thumbnail_size.y - scaled_height) / 2); - ImVec2 img_max = ImVec2(img_min.x + scaled_width, img_min.y + scaled_height); + ScaleDimensions(thumbnail_width, thumbnail_height, thumbnail_size.x, + thumbnail_size.y, &scaled_width, &scaled_height); + ImVec2 img_min = + ImVec2(thumbnail_min.x + (thumbnail_size.x - scaled_width) / 2, + thumbnail_min.y + (thumbnail_size.y - scaled_height) / 2); + ImVec2 img_max = + ImVec2(img_min.x + scaled_width, img_min.y + scaled_height); draw_list->AddImage((ImTextureID)(uint64_t)thumbnail, img_min, img_max); // Snapshot title ImGui::PushFont(g_font_mgr.m_menu_font_medium); - draw_list->AddText(ImVec2(p0.x + name_pos.x, p0.y + name_pos.y), IM_COL32(255, 255, 255, 255), snapshot->name); + draw_list->AddText(ImVec2(p0.x + name_pos.x, p0.y + name_pos.y), + IM_COL32(255, 255, 255, 255), snapshot->name); ImGui::PopFont(); // Snapshot XBE title name ImGui::PushFont(g_font_mgr.m_menu_font_small); - const char *title_name = data->xbe_title_name ? data->xbe_title_name : "(Unknown XBE Title Name)"; - draw_list->AddText(ImVec2(p0.x + title_pos.x, p0.y + title_pos.y), IM_COL32(255, 255, 255, 200), title_name); + const char *title_name = data->xbe_title_name ? data->xbe_title_name : + "(Unknown XBE Title Name)"; + draw_list->AddText(ImVec2(p0.x + title_pos.x, p0.y + title_pos.y), + IM_COL32(255, 255, 255, 200), title_name); // Snapshot date - g_autoptr(GDateTime) date = g_date_time_new_from_unix_local(snapshot->date_sec); + g_autoptr(GDateTime) date = + g_date_time_new_from_unix_local(snapshot->date_sec); char *date_buf = g_date_time_format(date, "%Y-%m-%d %H:%M:%S"); - draw_list->AddText(ImVec2(p0.x + date_pos.x, p0.y + date_pos.y), IM_COL32(255, 255, 255, 200), date_buf); + draw_list->AddText(ImVec2(p0.x + date_pos.x, p0.y + date_pos.y), + IM_COL32(255, 255, 255, 200), date_buf); g_free(date_buf); // Snapshot keyboard binding if (current_snapshot_binding != -1) { - char *binding_text = g_strdup_printf("Bound to F%d", current_snapshot_binding + 5); - draw_list->AddText(ImVec2(p0.x + binding_pos.x, p0.y + binding_pos.y), IM_COL32(255, 255, 255, 200), binding_text); + char *binding_text = + g_strdup_printf("Bound to F%d", current_snapshot_binding + 5); + draw_list->AddText(ImVec2(p0.x + binding_pos.x, p0.y + binding_pos.y), + IM_COL32(255, 255, 255, 200), binding_text); g_free(binding_text); } @@ -758,7 +962,7 @@ void MainMenuSnapshotsView::ClearSearch() int MainMenuSnapshotsView::OnSearchTextUpdate(ImGuiInputTextCallbackData *data) { GError *gerr = NULL; - MainMenuSnapshotsView *win = (MainMenuSnapshotsView*)data->UserData; + MainMenuSnapshotsView *win = (MainMenuSnapshotsView *)data->UserData; if (win->m_search_regex) { g_free(win->m_search_regex); @@ -770,7 +974,8 @@ int MainMenuSnapshotsView::OnSearchTextUpdate(ImGuiInputTextCallbackData *data) } char *buf = g_strdup_printf("(.*)%s(.*)", data->Buf); - win->m_search_regex = g_regex_new(buf, (GRegexCompileFlags)0, (GRegexMatchFlags)0, &gerr); + win->m_search_regex = + g_regex_new(buf, (GRegexCompileFlags)0, (GRegexMatchFlags)0, &gerr); g_free(buf); if (gerr) { win->m_search_regex = NULL; @@ -785,14 +990,17 @@ void MainMenuSnapshotsView::Draw() g_snapshot_mgr.Refresh(); SectionTitle("Snapshots"); - Toggle("Filter by current title", &g_config.general.snapshots.filter_current_game, - "Only display snapshots created while running the currently running XBE"); + Toggle("Filter by current title", + &g_config.general.snapshots.filter_current_game, + "Only display snapshots created while running the currently running " + "XBE"); if (g_config.general.snapshots.filter_current_game) { struct xbe *xbe = xemu_get_xbe_info(); if (xbe && xbe->cert) { if (xbe->cert->m_titleid != m_current_title_id) { - char *title_name = g_utf16_to_utf8(xbe->cert->m_title_name, 40, NULL, NULL, NULL); + char *title_name = g_utf16_to_utf8(xbe->cert->m_title_name, 40, + NULL, NULL, NULL); if (title_name) { m_current_title_name = title_name; g_free(title_name); @@ -816,7 +1024,8 @@ void MainMenuSnapshotsView::Draw() bool snapshot_with_create_name_exists = false; for (int i = 0; i < g_snapshot_mgr.m_snapshots_len; ++i) { - if (g_strcmp0(m_search_buf.c_str(), g_snapshot_mgr.m_snapshots[i].name) == 0) { + if (g_strcmp0(m_search_buf.c_str(), + g_snapshot_mgr.m_snapshots[i].name) == 0) { snapshot_with_create_name_exists = true; break; } @@ -828,8 +1037,10 @@ void MainMenuSnapshotsView::Draw() ImGui::PushStyleColor(ImGuiCol_ButtonHovered, ImVec4(1, 0, 0, 1)); ImGui::PushStyleColor(ImGuiCol_ButtonActive, ImVec4(1, 0, 0, 1)); } - if (ImGui::Button(snapshot_with_create_name_exists ? "Replace" : "Create", ImVec2(-FLT_MIN, 0))) { - xemu_snapshots_save(m_search_buf.empty() ? NULL : m_search_buf.c_str(), NULL); + if (ImGui::Button(snapshot_with_create_name_exists ? "Replace" : "Create", + ImVec2(-FLT_MIN, 0))) { + xemu_snapshots_save(m_search_buf.empty() ? NULL : m_search_buf.c_str(), + NULL); ClearSearch(); } if (snapshot_with_create_name_exists) { @@ -837,28 +1048,36 @@ void MainMenuSnapshotsView::Draw() } if (snapshot_with_create_name_exists && ImGui::IsItemHovered()) { - ImGui::SetTooltip("A snapshot with the name \"%s\" already exists. This button will overwrite the existing snapshot.", m_search_buf.c_str()); + ImGui::SetTooltip("A snapshot with the name \"%s\" already exists. " + "This button will overwrite the existing snapshot.", + m_search_buf.c_str()); } ImGui::PopFont(); bool at_least_one_snapshot_displayed = false; for (int i = g_snapshot_mgr.m_snapshots_len - 1; i >= 0; i--) { - if (g_config.general.snapshots.filter_current_game && g_snapshot_mgr.m_extra_data[i].xbe_title_name && - m_current_title_name.size() && strcmp(m_current_title_name.c_str(), g_snapshot_mgr.m_extra_data[i].xbe_title_name)) { + if (g_config.general.snapshots.filter_current_game && + g_snapshot_mgr.m_extra_data[i].xbe_title_name && + m_current_title_name.size() && + strcmp(m_current_title_name.c_str(), + g_snapshot_mgr.m_extra_data[i].xbe_title_name)) { continue; } if (m_search_regex) { GMatchInfo *match; bool keep_entry = false; - - g_regex_match(m_search_regex, g_snapshot_mgr.m_snapshots[i].name, (GRegexMatchFlags)0, &match); + + g_regex_match(m_search_regex, g_snapshot_mgr.m_snapshots[i].name, + (GRegexMatchFlags)0, &match); keep_entry |= g_match_info_matches(match); g_match_info_free(match); if (g_snapshot_mgr.m_extra_data[i].xbe_title_name) { - g_regex_match(m_search_regex, g_snapshot_mgr.m_extra_data[i].xbe_title_name, (GRegexMatchFlags)0, &match); + g_regex_match(m_search_regex, + g_snapshot_mgr.m_extra_data[i].xbe_title_name, + (GRegexMatchFlags)0, &match); keep_entry |= g_match_info_matches(match); g_free(match); } @@ -873,7 +1092,8 @@ void MainMenuSnapshotsView::Draw() int current_snapshot_binding = -1; for (int i = 0; i < 4; ++i) { - if (g_strcmp0(*(g_snapshot_shortcut_index_key_map[i]), snapshot->name) == 0) { + if (g_strcmp0(*(g_snapshot_shortcut_index_key_map[i]), + snapshot->name) == 0) { assert(current_snapshot_binding == -1); current_snapshot_binding = i; } @@ -885,13 +1105,14 @@ void MainMenuSnapshotsView::Draw() bool load = BigSnapshotButton(snapshot, data, current_snapshot_binding); // FIXME: Provide context menu control annotation - if (ImGui::IsItemHovered() && ImGui::IsKeyPressed(ImGuiKey_GamepadFaceLeft)) { + if (ImGui::IsItemHovered() && + ImGui::IsKeyPressed(ImGuiKey_GamepadFaceLeft)) { ImGui::SetNextWindowPos(pos); ImGui::OpenPopup("Snapshot Options"); } - + DrawSnapshotContextMenu(snapshot, data, current_snapshot_binding); - + ImGui::PopID(); if (load) { @@ -915,12 +1136,14 @@ void MainMenuSnapshotsView::Draw() } ImVec2 dim = ImGui::CalcTextSize(msg); ImVec2 cur = ImGui::GetCursorPos(); - ImGui::SetCursorPosX(cur.x + (ImGui::GetColumnWidth()-dim.x)/2); + ImGui::SetCursorPosX(cur.x + (ImGui::GetColumnWidth() - dim.x) / 2); ImGui::TextColored(ImVec4(0.94f, 0.94f, 0.94f, 0.70f), "%s", msg); } } -void MainMenuSnapshotsView::DrawSnapshotContextMenu(QEMUSnapshotInfo *snapshot, XemuSnapshotData *data, int current_snapshot_binding) +void MainMenuSnapshotsView::DrawSnapshotContextMenu( + QEMUSnapshotInfo *snapshot, XemuSnapshotData *data, + int current_snapshot_binding) { if (!ImGui::BeginPopupContextItem("Snapshot Options")) { return; @@ -936,9 +1159,12 @@ void MainMenuSnapshotsView::DrawSnapshotContextMenu(QEMUSnapshotInfo *snapshot, if (ImGui::MenuItem(item_name)) { if (current_snapshot_binding >= 0) { - xemu_settings_set_string(g_snapshot_shortcut_index_key_map[current_snapshot_binding], ""); + xemu_settings_set_string(g_snapshot_shortcut_index_key_map + [current_snapshot_binding], + ""); } - xemu_settings_set_string(g_snapshot_shortcut_index_key_map[i], snapshot->name); + xemu_settings_set_string(g_snapshot_shortcut_index_key_map[i], + snapshot->name); current_snapshot_binding = i; ImGui::CloseCurrentPopup(); @@ -949,13 +1175,15 @@ void MainMenuSnapshotsView::DrawSnapshotContextMenu(QEMUSnapshotInfo *snapshot, if (current_snapshot_binding >= 0) { if (ImGui::MenuItem("Unbind")) { - xemu_settings_set_string(g_snapshot_shortcut_index_key_map[current_snapshot_binding], ""); + xemu_settings_set_string( + g_snapshot_shortcut_index_key_map[current_snapshot_binding], + ""); current_snapshot_binding = -1; } } ImGui::EndMenu(); } - + ImGui::Separator(); Error *err = NULL; @@ -982,18 +1210,25 @@ MainMenuSystemView::MainMenuSystemView() : m_dirty(false) void MainMenuSystemView::Draw() { - const char *rom_file_filters = ".bin Files\0*.bin\0.rom Files\0*.rom\0All Files\0*.*\0"; + const char *rom_file_filters = + ".bin Files\0*.bin\0.rom Files\0*.rom\0All Files\0*.*\0"; const char *qcow_file_filters = ".qcow2 Files\0*.qcow2\0All Files\0*.*\0"; if (m_dirty) { - ImGui::TextColored(ImVec4(1,0,0,1), "Application restart required to apply settings"); + ImGui::TextColored(ImVec4(1, 0, 0, 1), + "Application restart required to apply settings"); + } + + if ((int)g_config.sys.avpack == CONFIG_SYS_AVPACK_NONE) { + ImGui::TextColored(ImVec4(1,0,0,1), "Setting AV Pack to NONE disables video output."); } SectionTitle("System Configuration"); if (ChevronCombo( "System Memory", &g_config.sys.mem_limit, - "64 MiB (Default)\0""128 MiB\0", + "64 MiB (Default)\0" + "128 MiB\0", "Increase to 128 MiB for debug or homebrew applications")) { m_dirty = true; } @@ -1026,8 +1261,9 @@ void MainMenuSystemView::Draw() } } -MainMenuAboutView::MainMenuAboutView(): m_config_info_text{NULL} -{} +MainMenuAboutView::MainMenuAboutView() : m_config_info_text{ NULL } +{ +} void MainMenuAboutView::UpdateConfigInfoText() { @@ -1035,20 +1271,21 @@ void MainMenuAboutView::UpdateConfigInfoText() g_free(m_config_info_text); } - gchar *bootrom_checksum = GetFileMD5Checksum(g_config.sys.files.bootrom_path); + gchar *bootrom_checksum = + GetFileMD5Checksum(g_config.sys.files.bootrom_path); if (!bootrom_checksum) { bootrom_checksum = g_strdup("None"); } - gchar *flash_rom_checksum = GetFileMD5Checksum(g_config.sys.files.flashrom_path); + gchar *flash_rom_checksum = + GetFileMD5Checksum(g_config.sys.files.flashrom_path); if (!flash_rom_checksum) { flash_rom_checksum = g_strdup("None"); } - m_config_info_text = g_strdup_printf( - "MCPX Boot ROM MD5 Hash: %s\n" - "Flash ROM (BIOS) MD5 Hash: %s", - bootrom_checksum, flash_rom_checksum); + m_config_info_text = g_strdup_printf("MCPX Boot ROM MD5 Hash: %s\n" + "Flash ROM (BIOS) MD5 Hash: %s", + bootrom_checksum, flash_rom_checksum); g_free(bootrom_checksum); g_free(flash_rom_checksum); } @@ -1057,22 +1294,25 @@ void MainMenuAboutView::Draw() { static const char *build_info_text = NULL; if (build_info_text == NULL) { - build_info_text = g_strdup_printf( - "Version: %s\nBranch: %s\nCommit: %s\nDate: %s", - xemu_version, xemu_branch, xemu_commit, xemu_date); + build_info_text = + g_strdup_printf("Version: %s\nBranch: %s\nCommit: " + "%s\nDate: %s", + xemu_version, xemu_branch, xemu_commit, xemu_date); } static const char *sys_info_text = NULL; if (sys_info_text == NULL) { - const char *gl_shader_version = (const char*)glGetString(GL_SHADING_LANGUAGE_VERSION); - const char *gl_version = (const char*)glGetString(GL_VERSION); - const char *gl_renderer = (const char*)glGetString(GL_RENDERER); - const char *gl_vendor = (const char*)glGetString(GL_VENDOR); + const char *gl_shader_version = + (const char *)glGetString(GL_SHADING_LANGUAGE_VERSION); + const char *gl_version = (const char *)glGetString(GL_VERSION); + const char *gl_renderer = (const char *)glGetString(GL_RENDERER); + const char *gl_vendor = (const char *)glGetString(GL_VENDOR); sys_info_text = g_strdup_printf( - "CPU: %s\nOS Platform: %s\nOS Version: %s\nManufacturer: %s\n" + "CPU: %s\nOS Platform: %s\nOS Version: " + "%s\nManufacturer: %s\n" "GPU Model: %s\nDriver: %s\nShader: %s", - xemu_get_cpu_info(), xemu_get_os_platform(), xemu_get_os_info(), gl_vendor, - gl_renderer, gl_version, gl_shader_version); + xemu_get_cpu_info(), xemu_get_os_platform(), xemu_get_os_info(), + gl_vendor, gl_renderer, gl_version, gl_shader_version); } if (m_config_info_text == NULL) { @@ -1117,7 +1357,7 @@ void MainMenuAboutView::Draw() } MainMenuTabButton::MainMenuTabButton(std::string text, std::string icon) -: m_icon(icon), m_text(text) + : m_icon(icon), m_text(text) { } @@ -1130,8 +1370,10 @@ bool MainMenuTabButton::Draw(bool selected) IM_COL32(0, 0, 0, 0); ImGui::PushStyleColor(ImGuiCol_Button, col); - ImGui::PushStyleColor(ImGuiCol_ButtonHovered, selected ? col : IM_COL32(32, 32, 32, 255)); - ImGui::PushStyleColor(ImGuiCol_ButtonActive, selected ? col : IM_COL32(32, 32, 32, 255)); + ImGui::PushStyleColor(ImGuiCol_ButtonHovered, + selected ? col : IM_COL32(32, 32, 32, 255)); + ImGui::PushStyleColor(ImGuiCol_ButtonActive, + selected ? col : IM_COL32(32, 32, 32, 255)); int p = ImGui::GetTextLineHeight() * 0.5; ImGui::PushStyleVar(ImGuiStyleVar_FramePadding, ImVec2(p, p)); ImGui::PushStyleVar(ImGuiStyleVar_FrameRounding, 0); @@ -1150,15 +1392,14 @@ bool MainMenuTabButton::Draw(bool selected) } MainMenuScene::MainMenuScene() -: m_animation(0.12, 0.12), - m_general_button("General", ICON_FA_GEARS), - m_input_button("Input", ICON_FA_GAMEPAD), - m_display_button("Display", ICON_FA_TV), - m_audio_button("Audio", ICON_FA_VOLUME_HIGH), - m_network_button("Network", ICON_FA_NETWORK_WIRED), - m_snapshots_button("Snapshots", ICON_FA_CLOCK_ROTATE_LEFT), - m_system_button("System", ICON_FA_MICROCHIP), - m_about_button("About", ICON_FA_CIRCLE_INFO) + : m_animation(0.12, 0.12), m_general_button("General", ICON_FA_GEARS), + m_input_button("Input", ICON_FA_GAMEPAD), + m_display_button("Display", ICON_FA_TV), + m_audio_button("Audio", ICON_FA_VOLUME_HIGH), + m_network_button("Network", ICON_FA_NETWORK_WIRED), + m_snapshots_button("Snapshots", ICON_FA_CLOCK_ROTATE_LEFT), + m_system_button("System", ICON_FA_MICROCHIP), + m_about_button("About", ICON_FA_CIRCLE_INFO) { m_had_focus_last_frame = false; m_focus_view = false; @@ -1184,34 +1425,21 @@ MainMenuScene::MainMenuScene() m_next_view_index = m_current_view_index; } -void MainMenuScene::ShowGeneral() -{ - SetNextViewIndexWithFocus(0); -} -void MainMenuScene::ShowInput() -{ - SetNextViewIndexWithFocus(1); -} -void MainMenuScene::ShowDisplay() -{ - SetNextViewIndexWithFocus(2); -} -void MainMenuScene::ShowAudio() -{ - SetNextViewIndexWithFocus(3); -} -void MainMenuScene::ShowNetwork() +void MainMenuScene::ShowSettings() { - SetNextViewIndexWithFocus(4); + SetNextViewIndexWithFocus(g_config.general.last_viewed_menu_index); } -void MainMenuScene::ShowSnapshots() + +void MainMenuScene::ShowSnapshots() { SetNextViewIndexWithFocus(5); } + void MainMenuScene::ShowSystem() { SetNextViewIndexWithFocus(6); } + void MainMenuScene::ShowAbout() { SetNextViewIndexWithFocus(7); @@ -1258,11 +1486,12 @@ void MainMenuScene::HandleInput() bool focus = ImGui::IsWindowFocused(ImGuiFocusedFlags_RootAndChildWindows | ImGuiFocusedFlags_NoPopupHierarchy); - // XXX: Ensure we have focus for two frames. If a user cancels a popup window, we do not want to cancel main + // XXX: Ensure we have focus for two frames. If a user cancels a popup + // window, we do not want to cancel main // window as well. if (nofocus || (focus && m_had_focus_last_frame && - (ImGui::IsKeyDown(ImGuiKey_GamepadFaceRight) - || ImGui::IsKeyDown(ImGuiKey_Escape)))) { + (ImGui::IsKeyDown(ImGuiKey_GamepadFaceRight) || + ImGui::IsKeyDown(ImGuiKey_Escape)))) { Hide(); return; } @@ -1322,9 +1551,10 @@ bool MainMenuScene::Draw() float nav_width = width * 0.3; float content_width = width - nav_width; - ImGui::PushStyleColor(ImGuiCol_ChildBg, IM_COL32(26,26,26,255)); + ImGui::PushStyleColor(ImGuiCol_ChildBg, IM_COL32(26, 26, 26, 255)); - ImGui::BeginChild("###MainWindowNav", ImVec2(nav_width, -1), true, ImGuiWindowFlags_NavFlattened); + ImGui::BeginChild("###MainWindowNav", ImVec2(nav_width, -1), true, + ImGuiWindowFlags_NavFlattened); bool move_focus_to_tab = false; if (m_current_view_index != m_next_view_index) { @@ -1358,7 +1588,8 @@ bool MainMenuScene::Draw() int s = ImGui::GetTextLineHeight() * 0.75; ImGui::PushStyleVar(ImGuiStyleVar_WindowPadding, ImVec2(s, s)); ImGui::PushStyleVar(ImGuiStyleVar_ItemSpacing, ImVec2(s, s)); - ImGui::PushStyleVar(ImGuiStyleVar_FrameRounding, 6*g_viewport_mgr.m_scale); + ImGui::PushStyleVar(ImGuiStyleVar_FrameRounding, + 6 * g_viewport_mgr.m_scale); ImGui::PushID(m_current_view_index); ImGui::BeginChild("###MainWindowContent", ImVec2(content_width, -1), @@ -1373,7 +1604,9 @@ bool MainMenuScene::Draw() ImGui::PushStyleColor(ImGuiCol_Text, IM_COL32(255, 255, 255, 128)); ImGui::PushStyleColor(ImGuiCol_Button, IM_COL32_BLACK_TRANS); ImVec2 pos = ImGui::GetCursorPos(); - ImGui::SetCursorPosX(ImGui::GetContentRegionMax().x - style.FramePadding.x * 2.0f - ImGui::GetTextLineHeight()); + ImGui::SetCursorPosX(ImGui::GetContentRegionMax().x - + style.FramePadding.x * 2.0f - + ImGui::GetTextLineHeight()); if (ImGui::Button(ICON_FA_XMARK)) { Hide(); } diff --git a/ui/xui/main-menu.hh b/ui/xui/main-menu.hh index 7be564701ca..bce3927333c 100644 --- a/ui/xui/main-menu.hh +++ b/ui/xui/main-menu.hh @@ -182,14 +182,10 @@ protected: public: MainMenuScene(); - void ShowGeneral(); - void ShowInput(); - void ShowDisplay(); - void ShowAudio(); - void ShowNetwork(); - void ShowSnapshots(); + void ShowSettings(); void ShowSystem(); void ShowAbout(); + void ShowSnapshots(); void SetNextViewIndexWithFocus(int i); void Show() override; void Hide() override; diff --git a/ui/xui/menubar.cc b/ui/xui/menubar.cc index 9ce8a4a0c7c..2d1f48c6045 100644 --- a/ui/xui/menubar.cc +++ b/ui/xui/menubar.cc @@ -120,7 +120,7 @@ void ShowMainMenu() } if (ImGui::MenuItem(save_name, hotkey, false, bound)) { - ActionActivateBoundSnapshot(i, false); + ActionActivateBoundSnapshot(i, true); } g_free(hotkey); @@ -138,14 +138,7 @@ void ShowMainMenu() ImGui::Separator(); - ImGui::MenuItem("Settings", NULL, false, false); - if (ImGui::MenuItem(" General")) g_main_menu.ShowGeneral(); - if (ImGui::MenuItem(" Input")) g_main_menu.ShowInput(); - if (ImGui::MenuItem(" Display")) g_main_menu.ShowDisplay(); - if (ImGui::MenuItem(" Audio")) g_main_menu.ShowAudio(); - if (ImGui::MenuItem(" Network")) g_main_menu.ShowNetwork(); - if (ImGui::MenuItem(" Snapshots")) g_main_menu.ShowSnapshots(); - if (ImGui::MenuItem(" System")) g_main_menu.ShowSystem(); + if (ImGui::MenuItem("Settings...")) g_main_menu.ShowSettings(); ImGui::Separator(); diff --git a/ui/xui/widgets.cc b/ui/xui/widgets.cc index 34a122ce6c3..67430ecda8a 100644 --- a/ui/xui/widgets.cc +++ b/ui/xui/widgets.cc @@ -312,7 +312,8 @@ bool FilePicker(const char *str_id, const char **buf, const char *filters, GetWidgetTitleDescriptionHeight(str_id, desc)); ImGui::PushStyleVar(ImGuiStyleVar_ButtonTextAlign, ImVec2(0, 0)); ImGui::PushID(str_id); - bool status = ImGui::Button("###file_button", bb); + bool status = + ImGui::ButtonEx("###file_button", bb, ImGuiButtonFlags_AllowOverlap); ImGui::SetItemAllowOverlap(); if (status) { int flags = NOC_FILE_DIALOG_OPEN;