2016-03-30 23:07:55 +03:00
|
|
|
#include "gb.h"
|
2017-04-17 20:16:17 +03:00
|
|
|
#include <assert.h>
|
2016-03-30 23:07:55 +03:00
|
|
|
|
2016-06-18 20:29:11 +03:00
|
|
|
void GB_update_joyp(GB_gameboy_t *gb)
|
2016-03-30 23:07:55 +03:00
|
|
|
{
|
2019-07-19 20:27:53 +03:00
|
|
|
if (gb->model & GB_MODEL_NO_SFC_BIT) return;
|
2019-07-16 21:42:57 +03:00
|
|
|
|
2016-06-18 20:29:11 +03:00
|
|
|
uint8_t key_selection = 0;
|
|
|
|
uint8_t previous_state = 0;
|
2016-03-30 23:07:55 +03:00
|
|
|
|
|
|
|
/* Todo: add delay to key selection */
|
|
|
|
previous_state = gb->io_registers[GB_IO_JOYP] & 0xF;
|
|
|
|
key_selection = (gb->io_registers[GB_IO_JOYP] >> 4) & 3;
|
|
|
|
gb->io_registers[GB_IO_JOYP] &= 0xF0;
|
2021-10-09 14:52:28 +03:00
|
|
|
uint8_t current_player = gb->sgb? gb->sgb->current_player : 0;
|
2016-03-30 23:07:55 +03:00
|
|
|
switch (key_selection) {
|
|
|
|
case 3:
|
2018-11-16 01:53:01 +02:00
|
|
|
if (gb->sgb && gb->sgb->player_count > 1) {
|
|
|
|
gb->io_registers[GB_IO_JOYP] |= 0xF - current_player;
|
2018-11-12 00:37:06 +02:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
/* Nothing is wired, all up */
|
|
|
|
gb->io_registers[GB_IO_JOYP] |= 0x0F;
|
|
|
|
}
|
2016-03-30 23:07:55 +03:00
|
|
|
break;
|
|
|
|
|
|
|
|
case 2:
|
|
|
|
/* Direction keys */
|
2016-06-18 20:29:11 +03:00
|
|
|
for (uint8_t i = 0; i < 4; i++) {
|
2018-11-16 01:53:01 +02:00
|
|
|
gb->io_registers[GB_IO_JOYP] |= (!gb->keys[current_player][i]) << i;
|
2016-03-30 23:07:55 +03:00
|
|
|
}
|
2016-09-21 02:15:02 +03:00
|
|
|
/* Forbid pressing two opposing keys, this breaks a lot of games; even if it's somewhat possible. */
|
2021-12-19 00:28:24 +02:00
|
|
|
if (likely(!gb->illegal_inputs_allowed)) {
|
|
|
|
if (!(gb->io_registers[GB_IO_JOYP] & 1)) {
|
|
|
|
gb->io_registers[GB_IO_JOYP] |= 2;
|
|
|
|
}
|
|
|
|
if (!(gb->io_registers[GB_IO_JOYP] & 4)) {
|
|
|
|
gb->io_registers[GB_IO_JOYP] |= 8;
|
|
|
|
}
|
2016-09-21 02:15:02 +03:00
|
|
|
}
|
2016-03-30 23:07:55 +03:00
|
|
|
break;
|
|
|
|
|
|
|
|
case 1:
|
|
|
|
/* Other keys */
|
2016-06-18 20:29:11 +03:00
|
|
|
for (uint8_t i = 0; i < 4; i++) {
|
2018-11-16 01:53:01 +02:00
|
|
|
gb->io_registers[GB_IO_JOYP] |= (!gb->keys[current_player][i + 4]) << i;
|
2016-03-30 23:07:55 +03:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0:
|
2016-06-18 20:29:11 +03:00
|
|
|
for (uint8_t i = 0; i < 4; i++) {
|
2018-11-16 01:53:01 +02:00
|
|
|
gb->io_registers[GB_IO_JOYP] |= (!(gb->keys[current_player][i] || gb->keys[current_player][i + 4])) << i;
|
2016-03-30 23:07:55 +03:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2022-01-03 17:17:35 +02:00
|
|
|
nodefault;
|
2016-03-30 23:07:55 +03:00
|
|
|
}
|
2018-11-11 01:16:32 +02:00
|
|
|
|
2019-07-16 21:42:57 +03:00
|
|
|
/* Todo: This assumes the keys *always* bounce, which is incorrect when emulating an SGB */
|
2016-03-30 23:07:55 +03:00
|
|
|
if (previous_state != (gb->io_registers[GB_IO_JOYP] & 0xF)) {
|
2019-07-16 21:42:57 +03:00
|
|
|
/* The joypad interrupt DOES occur on CGB (Tested on CGB-E), unlike what some documents say. */
|
2022-01-09 16:43:32 +02:00
|
|
|
if (!(gb->io_registers[GB_IO_IF] & 0x10)) {
|
|
|
|
gb->joyp_accessed = true;
|
|
|
|
gb->io_registers[GB_IO_IF] |= 0x10;
|
|
|
|
}
|
2016-03-30 23:07:55 +03:00
|
|
|
}
|
2018-11-11 01:16:32 +02:00
|
|
|
|
|
|
|
gb->io_registers[GB_IO_JOYP] |= 0xC0;
|
2017-04-17 20:16:17 +03:00
|
|
|
}
|
|
|
|
|
2019-07-16 21:42:57 +03:00
|
|
|
void GB_icd_set_joyp(GB_gameboy_t *gb, uint8_t value)
|
|
|
|
{
|
|
|
|
uint8_t previous_state = gb->io_registers[GB_IO_JOYP] & 0xF;
|
|
|
|
gb->io_registers[GB_IO_JOYP] &= 0xF0;
|
|
|
|
gb->io_registers[GB_IO_JOYP] |= value & 0xF;
|
|
|
|
|
|
|
|
if (previous_state & ~(gb->io_registers[GB_IO_JOYP] & 0xF)) {
|
2022-01-09 16:43:32 +02:00
|
|
|
if (!(gb->io_registers[GB_IO_IF] & 0x10)) {
|
|
|
|
gb->joyp_accessed = true;
|
|
|
|
gb->io_registers[GB_IO_IF] |= 0x10;
|
|
|
|
}
|
2019-07-16 21:42:57 +03:00
|
|
|
}
|
2019-09-13 13:13:28 +03:00
|
|
|
gb->io_registers[GB_IO_JOYP] |= 0xC0;
|
2019-07-16 21:42:57 +03:00
|
|
|
}
|
|
|
|
|
2017-04-17 20:16:17 +03:00
|
|
|
void GB_set_key_state(GB_gameboy_t *gb, GB_key_t index, bool pressed)
|
|
|
|
{
|
|
|
|
assert(index >= 0 && index < GB_KEY_MAX);
|
2018-11-12 00:37:06 +02:00
|
|
|
gb->keys[0][index] = pressed;
|
2019-06-07 18:27:25 +03:00
|
|
|
GB_update_joyp(gb);
|
2018-11-12 00:37:06 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void GB_set_key_state_for_player(GB_gameboy_t *gb, GB_key_t index, unsigned player, bool pressed)
|
|
|
|
{
|
|
|
|
assert(index >= 0 && index < GB_KEY_MAX);
|
|
|
|
assert(player < 4);
|
|
|
|
gb->keys[player][index] = pressed;
|
2019-06-07 18:27:25 +03:00
|
|
|
GB_update_joyp(gb);
|
2017-04-17 20:16:17 +03:00
|
|
|
}
|
2021-12-02 11:23:44 +02:00
|
|
|
|
|
|
|
void GB_set_key_mask(GB_gameboy_t *gb, GB_key_mask_t mask)
|
|
|
|
{
|
|
|
|
memset(gb->keys, 0, sizeof(gb->keys));
|
|
|
|
bool *key = &gb->keys[0][0];
|
|
|
|
while (mask) {
|
|
|
|
if (mask & 1) {
|
|
|
|
*key = true;
|
|
|
|
}
|
|
|
|
mask >>= 1;
|
2021-12-02 11:54:26 +02:00
|
|
|
key++;
|
2021-12-02 11:23:44 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
GB_update_joyp(gb);
|
|
|
|
}
|
|
|
|
|
|
|
|
void GB_set_key_mask_for_player(GB_gameboy_t *gb, GB_key_mask_t mask, unsigned player)
|
|
|
|
{
|
|
|
|
memset(gb->keys[player], 0, sizeof(gb->keys[player]));
|
|
|
|
bool *key = gb->keys[player];
|
|
|
|
while (mask) {
|
|
|
|
if (mask & 1) {
|
|
|
|
*key = true;
|
|
|
|
}
|
|
|
|
mask >>= 1;
|
2021-12-02 11:54:26 +02:00
|
|
|
key++;
|
2021-12-02 11:23:44 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
GB_update_joyp(gb);
|
|
|
|
}
|
2021-12-18 01:25:06 +02:00
|
|
|
|
|
|
|
bool GB_get_joyp_accessed(GB_gameboy_t *gb)
|
|
|
|
{
|
|
|
|
return gb->joyp_accessed;
|
|
|
|
}
|
|
|
|
|
|
|
|
void GB_clear_joyp_accessed(GB_gameboy_t *gb)
|
|
|
|
{
|
|
|
|
gb->joyp_accessed = false;
|
|
|
|
}
|
2021-12-19 00:28:24 +02:00
|
|
|
|
|
|
|
void GB_set_allow_illegal_inputs(GB_gameboy_t *gb, bool allow)
|
|
|
|
{
|
|
|
|
gb->illegal_inputs_allowed = allow;
|
|
|
|
}
|