206 lines
5.1 KiB
C
206 lines
5.1 KiB
C
// MIT License
|
|
|
|
// Copyright (c) 2017 Vadim Grigoruk @nesbox
|
|
|
|
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
// of this software and associated documentation files (the "Software"), to deal
|
|
// in the Software without restriction, including without limitation the rights
|
|
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
// copies of the Software, and to permit persons to whom the Software is
|
|
// furnished to do so, subject to the following conditions:
|
|
|
|
// The above copyright notice and this permission notice shall be included in all
|
|
// copies or substantial portions of the Software.
|
|
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
// SOFTWARE.
|
|
|
|
#include <stdio.h>
|
|
#include <SDL.h>
|
|
#include <tic80.h>
|
|
|
|
// TODO: take from tic.h??
|
|
#define TIC_FRAMERATE 60
|
|
|
|
static struct
|
|
{
|
|
bool quit;
|
|
} state =
|
|
{
|
|
.quit = false,
|
|
};
|
|
|
|
static void onExit()
|
|
{
|
|
state.quit = true;
|
|
}
|
|
|
|
int main(int argc, char **argv)
|
|
{
|
|
FILE* file = fopen("cart.tic", "rb");
|
|
|
|
if(file)
|
|
{
|
|
fseek(file, 0, SEEK_END);
|
|
int size = ftell(file);
|
|
fseek(file, 0, SEEK_SET);
|
|
|
|
void* cart = SDL_malloc(size);
|
|
if(cart) fread(cart, size, 1, file);
|
|
fclose(file);
|
|
|
|
if(cart)
|
|
{
|
|
SDL_Init(SDL_INIT_VIDEO | SDL_INIT_AUDIO);
|
|
|
|
{
|
|
SDL_Window* window = SDL_CreateWindow("TIC-80 SDL demo", SDL_WINDOWPOS_CENTERED, SDL_WINDOWPOS_CENTERED, TIC80_FULLWIDTH, TIC80_FULLHEIGHT, SDL_WINDOW_SHOWN | SDL_WINDOW_RESIZABLE);
|
|
SDL_Renderer* renderer = SDL_CreateRenderer(window, -1, SDL_RENDERER_ACCELERATED);
|
|
SDL_Texture* texture = SDL_CreateTexture(renderer, SDL_PIXELFORMAT_ABGR8888, SDL_TEXTUREACCESS_STREAMING, TIC80_FULLWIDTH, TIC80_FULLHEIGHT);
|
|
|
|
SDL_AudioDeviceID audioDevice = 0;
|
|
SDL_AudioSpec audioSpec;
|
|
SDL_AudioCVT cvt;
|
|
bool audioStarted = false;
|
|
|
|
{
|
|
SDL_AudioSpec want =
|
|
{
|
|
.freq = 44100,
|
|
.format = AUDIO_S16,
|
|
.channels = 1,
|
|
.userdata = NULL,
|
|
};
|
|
|
|
audioDevice = SDL_OpenAudioDevice(NULL, 0, &want, &audioSpec, SDL_AUDIO_ALLOW_ANY_CHANGE);
|
|
|
|
SDL_BuildAudioCVT(&cvt, want.format, want.channels, audioSpec.freq, audioSpec.format, audioSpec.channels, audioSpec.freq);
|
|
|
|
if (cvt.needed)
|
|
{
|
|
cvt.len = audioSpec.freq * sizeof(s16) / TIC_FRAMERATE;
|
|
cvt.buf = SDL_malloc(cvt.len * cvt.len_mult);
|
|
}
|
|
}
|
|
|
|
tic80_input input;
|
|
SDL_memset(&input, 0, sizeof input);
|
|
|
|
tic80* tic = tic80_create(audioSpec.freq);
|
|
|
|
tic->callback.exit = onExit;
|
|
|
|
tic80_load(tic, cart, size);
|
|
|
|
if(tic)
|
|
{
|
|
u64 nextTick = SDL_GetPerformanceCounter();
|
|
const u64 Delta = SDL_GetPerformanceFrequency() / TIC_FRAMERATE;
|
|
|
|
while(!state.quit)
|
|
{
|
|
SDL_Event event;
|
|
|
|
while(SDL_PollEvent(&event))
|
|
{
|
|
switch(event.type)
|
|
{
|
|
case SDL_QUIT:
|
|
state.quit = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
{
|
|
input.gamepads.data = 0;
|
|
const uint8_t* keyboard = SDL_GetKeyboardState(NULL);
|
|
|
|
static const SDL_Scancode Keys[] =
|
|
{
|
|
SDL_SCANCODE_UP,
|
|
SDL_SCANCODE_DOWN,
|
|
SDL_SCANCODE_LEFT,
|
|
SDL_SCANCODE_RIGHT,
|
|
|
|
SDL_SCANCODE_Z,
|
|
SDL_SCANCODE_X,
|
|
SDL_SCANCODE_A,
|
|
SDL_SCANCODE_S,
|
|
};
|
|
|
|
for (int i = 0; i < SDL_arraysize(Keys); i++)
|
|
{
|
|
if (keyboard[Keys[i]])
|
|
{
|
|
input.gamepads.first.data |= (1 << i);
|
|
}
|
|
}
|
|
}
|
|
|
|
nextTick += Delta;
|
|
|
|
tic80_tick(tic, input);
|
|
|
|
if (!audioStarted && audioDevice)
|
|
{
|
|
audioStarted = true;
|
|
SDL_PauseAudioDevice(audioDevice, 0);
|
|
}
|
|
|
|
{
|
|
SDL_PauseAudioDevice(audioDevice, 0);
|
|
s32 size = tic->sound.count * sizeof(tic->sound.samples[0]);
|
|
|
|
if (cvt.needed)
|
|
{
|
|
SDL_memcpy(cvt.buf, tic->sound.samples, size);
|
|
SDL_ConvertAudio(&cvt);
|
|
SDL_QueueAudio(audioDevice, cvt.buf, cvt.len_cvt);
|
|
}
|
|
else SDL_QueueAudio(audioDevice, tic->sound.samples, size);
|
|
}
|
|
|
|
SDL_RenderClear(renderer);
|
|
|
|
{
|
|
void* pixels = NULL;
|
|
int pitch = 0;
|
|
SDL_LockTexture(texture, NULL, &pixels, &pitch);
|
|
SDL_memcpy(pixels, tic->screen, pitch * TIC80_FULLHEIGHT);
|
|
SDL_UnlockTexture(texture);
|
|
SDL_RenderCopy(renderer, texture, NULL, NULL);
|
|
}
|
|
|
|
SDL_RenderPresent(renderer);
|
|
|
|
{
|
|
s64 delay = nextTick - SDL_GetPerformanceCounter();
|
|
|
|
if (delay < 0)
|
|
nextTick -= delay;
|
|
else SDL_Delay((u32)(delay * 1000 / SDL_GetPerformanceFrequency()));
|
|
}
|
|
|
|
}
|
|
|
|
tic80_delete(tic);
|
|
}
|
|
|
|
SDL_DestroyTexture(texture);
|
|
SDL_DestroyRenderer(renderer);
|
|
SDL_DestroyWindow(window);
|
|
SDL_CloseAudioDevice(audioDevice);
|
|
}
|
|
|
|
SDL_free(cart);
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|