annotate examples/tank_wars/render.cpp @ 410:41436dcd2cb0 prerelease

Makefile can stop ide (idestop)
author Giulio Moro <giuliomoro@yahoo.it>
date Wed, 15 Jun 2016 20:58:48 +0100
parents 9dc5a0ccad25
children
rev   line source
andrewm@10 1 /*
andrewm@10 2 * render.cpp
andrewm@10 3 *
andrewm@10 4 * Created on: Oct 24, 2014
andrewm@10 5 * Author: parallels
andrewm@10 6 */
andrewm@10 7
andrewm@10 8
giuliomoro@301 9 #include <Bela.h>
andrewm@10 10 #include "game.h"
andrewm@10 11 #include <rtdk.h>
andrewm@10 12 #include <cmath>
andrewm@10 13 #include <cstdlib>
andrewm@10 14 #include <time.h>
andrewm@10 15
andrewm@22 16 int gAudioFramesPerMatrixFrame = 2; // Ratio in audio to matrix sample rate
andrewm@22 17
andrewm@10 18 int gInputTank1Angle = 0; // Inputs for the cannon angles
andrewm@10 19 int gInputTank2Angle = 1;
andrewm@10 20 int gInputLauncher = 2; // Input for launcher FSR
andrewm@10 21
andrewm@10 22 int gOutputX = 0; // Outputs for the scope
andrewm@10 23 int gOutputY = 1;
andrewm@10 24 int gOutputPlayer1LED = 2;
andrewm@10 25 int gOutputPlayer2LED = 3;
andrewm@10 26
andrewm@10 27 int gGameFramesPerSecond = 60; // How often the physics are updated
andrewm@10 28 int gGameFrameInterval; // ...and in frames
andrewm@10 29 int gSamplesUntilNextFrame; // Counter until next update
andrewm@10 30 int gSamplesSinceFinish = 0; // How long since somebody won?
andrewm@10 31 bool gGameShouldRestart = false;// Whether we need to reinitiliase the game
andrewm@10 32
andrewm@10 33 // Counter for overall number of samples that have elapsed
andrewm@10 34 unsigned int gSampleCounter = 0;
andrewm@10 35
andrewm@10 36 // 1st-order filter and peak detector for launcher input
andrewm@10 37 float gLauncherLastSample = 0;
andrewm@10 38 float gLauncherFilterPole = 0.8;
andrewm@10 39 float gLauncherPeakValue = 0;
andrewm@10 40 float gLauncherPeakFilterPole = 0.999;
andrewm@22 41 float gLauncherNoiseThreshold = 0.01;
andrewm@22 42 float gLauncherMinimumPeak = 0.1;
andrewm@10 43 bool gLauncherTriggered = false;
andrewm@10 44
andrewm@10 45 // Screen update rate; affects buffer size. Actual contents of buffer
andrewm@10 46 // may be smaller than this
andrewm@10 47 int gScreenWidth = 512;
andrewm@10 48 int gScreenHeight = 512;
andrewm@10 49 int gScreenFramesPerSecond = 25;
andrewm@10 50
andrewm@10 51 // Double buffer for rendering screen. Each is an interleaved buffer
andrewm@10 52 // of XY data.
andrewm@10 53 float *gScreenBuffer1, *gScreenBuffer2;
andrewm@10 54 float *gScreenBufferWrite, *gScreenBufferRead;
andrewm@10 55 int gScreenBufferMaxLength; // What is the total buffer allocated?
andrewm@10 56 int gScreenBufferReadLength; // How long is the read buffer?
andrewm@10 57 int gScreenBufferWriteLength; // How long is the write (next) buffer?
andrewm@10 58 int gScreenBufferReadPointer; // Where are we in the read buffer now?
andrewm@10 59 int gScreenBufferNextUpdateLocation; // When should we render the next buffer?
andrewm@10 60 bool gScreenNextBufferReady; // Is the next buffer ready to go?
andrewm@10 61
andrewm@10 62 // Auxiliary (low-priority) task for updating the screen
andrewm@10 63 AuxiliaryTask gScreenUpdateTask;
andrewm@10 64
andrewm@22 65 // Buffers for music and sound effects
andrewm@22 66 extern float *gMusicBuffer;
andrewm@22 67 extern int gMusicBufferLength;
andrewm@22 68 extern float *gSoundBoomBuffer;
andrewm@22 69 extern int gSoundBoomBufferLength;
andrewm@268 70 extern float *gSoundHitBuffer;
andrewm@268 71 extern int gSoundHitBufferLength;
andrewm@22 72
andrewm@22 73 // Current state for sound and music
andrewm@22 74 int gMusicBufferPointer = 0; // 0 means start of buffer...
andrewm@22 75 int gSoundBoomBufferPointer = -1; // -1 means don't play...
andrewm@268 76 int gSoundHitBufferPointer = -1;
andrewm@22 77 float gSoundProjectileOscillatorPhase = 0;
andrewm@22 78 float gSoundProjectileOscillatorGain = 0.2;
andrewm@22 79 float gOscillatorPhaseScaler = 0;
andrewm@22 80
andrewm@10 81 void screen_update();
andrewm@10 82
andrewm@56 83 // setup() is called once before the audio rendering starts.
andrewm@10 84 // Use it to perform any initialisation and allocation which is dependent
andrewm@10 85 // on the period size or sample rate.
andrewm@10 86 //
andrewm@10 87 // userData holds an opaque pointer to a data structure that was passed
andrewm@10 88 // in from the call to initAudio().
andrewm@10 89 //
andrewm@10 90 // Return true on success; returning false halts the program.
andrewm@10 91
giuliomoro@301 92 bool setup(BelaContext *context, void *userData)
andrewm@10 93 {
andrewm@10 94 srandom(time(NULL));
andrewm@10 95
andrewm@10 96 // Verify we are running with matrix enabled
andrewm@56 97 if(context->analogFrames == 0 || context->analogChannels < 4) {
andrewm@14 98 rt_printf("Error: this example needs the matrix enabled with at least 4 channels\n");
andrewm@10 99 return false;
andrewm@10 100 }
andrewm@10 101
andrewm@22 102 // Initialise audio variables
andrewm@56 103 gAudioFramesPerMatrixFrame = context->audioFrames / context->analogFrames;
andrewm@56 104 gOscillatorPhaseScaler = 2.0 * M_PI / context->audioSampleRate;
andrewm@22 105
andrewm@10 106 // Initialise the screen buffers
andrewm@56 107 gScreenBufferMaxLength = 2 * context->analogSampleRate / gScreenFramesPerSecond;
andrewm@10 108 gScreenBuffer1 = new float[gScreenBufferMaxLength];
andrewm@10 109 gScreenBuffer2 = new float[gScreenBufferMaxLength];
andrewm@10 110 if(gScreenBuffer1 == 0 || gScreenBuffer2 == 0) {
andrewm@10 111 rt_printf("Error initialising screen buffers\n");
andrewm@10 112 return false;
andrewm@10 113 }
andrewm@10 114
andrewm@10 115 gScreenBufferRead = gScreenBuffer1;
andrewm@10 116 gScreenBufferWrite = gScreenBuffer2;
andrewm@10 117 gScreenBufferReadLength = gScreenBufferWriteLength = 0;
andrewm@10 118 gScreenBufferReadPointer = 0;
andrewm@10 119 gScreenBufferNextUpdateLocation = 0;
andrewm@10 120 gScreenNextBufferReady = false;
andrewm@10 121
andrewm@10 122 // Initialise the game
andrewm@10 123 setupGame(gScreenWidth, gScreenHeight);
andrewm@56 124 gGameFrameInterval = context->analogSampleRate / gGameFramesPerSecond;
andrewm@10 125 gSamplesUntilNextFrame = gGameFrameInterval;
andrewm@10 126
andrewm@10 127 // Initialise auxiliary tasks
giuliomoro@301 128 if((gScreenUpdateTask = Bela_createAuxiliaryTask(&screen_update, 90,
andrewm@303 129 "bela-screen-update")) == 0)
andrewm@10 130 return false;
andrewm@10 131
andrewm@10 132 return true;
andrewm@10 133 }
andrewm@10 134
andrewm@10 135 // Swap buffers on the screen
andrewm@10 136 void swap_buffers()
andrewm@10 137 {
andrewm@10 138 if(gScreenBufferRead == gScreenBuffer1) {
andrewm@10 139 gScreenBufferRead = gScreenBuffer2;
andrewm@10 140 gScreenBufferWrite = gScreenBuffer1;
andrewm@10 141 }
andrewm@10 142 else {
andrewm@10 143 gScreenBufferRead = gScreenBuffer1;
andrewm@10 144 gScreenBufferWrite = gScreenBuffer2;
andrewm@10 145 }
andrewm@10 146
andrewm@10 147 gScreenBufferReadLength = gScreenBufferWriteLength;
andrewm@10 148 gScreenBufferReadPointer = 0;
andrewm@10 149
andrewm@10 150 // Schedule next update for 3/4 of the way through the buffer
andrewm@10 151 gScreenBufferNextUpdateLocation = gScreenBufferReadLength * 0.75;
andrewm@10 152 gScreenNextBufferReady = false;
andrewm@10 153 }
andrewm@10 154
andrewm@10 155 // render() is called regularly at the highest priority by the audio engine.
andrewm@10 156 // Input and output are given from the audio hardware and the other
andrewm@10 157 // ADCs and DACs (if available). If only audio is available, numMatrixFrames
andrewm@10 158 // will be 0.
andrewm@10 159
giuliomoro@301 160 void render(BelaContext *context, void *userData)
andrewm@10 161 {
andrewm@22 162 int audioIndex = 0;
andrewm@22 163
andrewm@56 164 for(unsigned int n = 0; n < context->analogFrames; n++) {
andrewm@22 165 for(int k = 0; k < gAudioFramesPerMatrixFrame; k++) {
andrewm@22 166 // Render music and sound
andrewm@22 167 float audioSample = 0;
andrewm@22 168
andrewm@22 169 // Music plays in a loop
andrewm@22 170 if(gMusicBuffer != 0 && gMusicBufferPointer >= 0) {
andrewm@22 171 audioSample += gMusicBuffer[gMusicBufferPointer++];
andrewm@22 172 if(gMusicBufferPointer >= gMusicBufferLength)
andrewm@22 173 gMusicBufferPointer = 0;
andrewm@22 174 }
andrewm@22 175
andrewm@22 176 // Sound effect plays until finished, then stops
andrewm@22 177 if(gSoundBoomBuffer != 0 && gSoundBoomBufferPointer >= 0) {
andrewm@22 178 audioSample += gSoundBoomBuffer[gSoundBoomBufferPointer++];
andrewm@22 179 if(gSoundBoomBufferPointer >= gSoundBoomBufferLength)
andrewm@22 180 gSoundBoomBufferPointer = -1;
andrewm@22 181 }
andrewm@22 182
andrewm@268 183 if(gSoundHitBuffer != 0 && gSoundHitBufferPointer >= 0) {
andrewm@268 184 audioSample += gSoundHitBuffer[gSoundHitBufferPointer++];
andrewm@268 185 if(gSoundHitBufferPointer >= gSoundHitBufferLength)
andrewm@268 186 gSoundHitBufferPointer = -1;
andrewm@268 187 }
andrewm@268 188
andrewm@22 189 // Oscillator plays to indicate projectile height
andrewm@22 190 if(gameStatusProjectileInMotion()) {
andrewm@22 191 audioSample += gSoundProjectileOscillatorGain * sinf(gSoundProjectileOscillatorPhase);
andrewm@22 192
andrewm@22 193 gSoundProjectileOscillatorPhase += gOscillatorPhaseScaler * constrain(map(gameStatusProjectileHeight(),
andrewm@22 194 1.0, 0, 300, 2000), 200, 6000);
andrewm@22 195 if(gSoundProjectileOscillatorPhase > 2.0 * M_PI)
andrewm@22 196 gSoundProjectileOscillatorPhase -= 2.0 * M_PI;
andrewm@22 197 }
andrewm@22 198
andrewm@56 199 context->audioOut[2*audioIndex] = context->audioOut[2*audioIndex + 1] = audioSample;
andrewm@22 200 audioIndex++;
andrewm@22 201 }
andrewm@22 202
andrewm@10 203 // First-order lowpass filter to remove noise on launch FSR
andrewm@311 204 float rawSample = analogRead(context, n, gInputLauncher);
andrewm@10 205 float launchSample = gLauncherFilterPole * gLauncherLastSample +
andrewm@10 206 (1.0f - gLauncherFilterPole) * rawSample;
andrewm@10 207 gLauncherLastSample = launchSample;
andrewm@10 208
andrewm@10 209 // Peak-detect on launch signal
andrewm@10 210 if(launchSample >= gLauncherPeakValue) {
andrewm@10 211 gLauncherPeakValue = launchSample;
andrewm@10 212 gLauncherTriggered = false;
andrewm@10 213 }
andrewm@10 214 else {
andrewm@10 215 if(gLauncherPeakValue - launchSample > gLauncherNoiseThreshold && !gLauncherTriggered) {
andrewm@10 216 // Detected a peak; is it big enough overall?
andrewm@10 217 if(gLauncherPeakValue >= gLauncherMinimumPeak) {
andrewm@10 218 gLauncherTriggered = true;
andrewm@10 219 // Peak detected-- fire!!
andrewm@10 220 // Set both cannon strengths but only one will
andrewm@10 221 // fire depending on whose turn it is
andrewm@10 222 float strength = map(gLauncherPeakValue,
andrewm@22 223 gLauncherMinimumPeak, 1.0,
andrewm@10 224 0.5f, 10.0f);
andrewm@10 225 setTank1CannonStrength(strength);
andrewm@10 226 setTank2CannonStrength(strength);
andrewm@10 227 fireProjectile();
andrewm@10 228 }
andrewm@10 229 }
andrewm@10 230
andrewm@10 231 gLauncherPeakValue *= gLauncherPeakFilterPole;
andrewm@10 232 }
andrewm@10 233
andrewm@10 234 if(--gSamplesUntilNextFrame <= 0) {
andrewm@10 235 // Update game physics and cannon angles
andrewm@10 236 gSamplesUntilNextFrame = gGameFrameInterval;
andrewm@10 237
andrewm@311 238 setTank1CannonAngle(map(analogRead(context, n, gInputTank1Angle),
andrewm@22 239 0, 1.0, M_PI, 0));
andrewm@311 240 setTank2CannonAngle(map(analogRead(context, n, gInputTank2Angle),
andrewm@22 241 0, 1.0, M_PI, 0));
andrewm@10 242 nextGameFrame();
andrewm@22 243
andrewm@22 244 // Check for collision and start sound accordingly
andrewm@22 245 if(gameStatusCollisionOccurred()) {
andrewm@22 246 gSoundBoomBufferPointer = 0;
andrewm@22 247 }
andrewm@268 248
andrewm@268 249 if(gameStatusTankHitOccurred()) {
andrewm@268 250 gSoundHitBufferPointer = 0;
andrewm@268 251 }
andrewm@10 252 }
andrewm@10 253
andrewm@10 254 if(gScreenBufferReadPointer >= gScreenBufferReadLength - 1
andrewm@10 255 && gScreenNextBufferReady) {
andrewm@10 256 // Got to the end; swap buffers
andrewm@10 257 swap_buffers();
andrewm@10 258 }
andrewm@10 259
andrewm@10 260 // Push current screen buffer to the matrix output
andrewm@10 261 if(gScreenBufferReadPointer < gScreenBufferReadLength - 1) {
andrewm@10 262 float x = gScreenBufferRead[gScreenBufferReadPointer++];
andrewm@10 263 float y = gScreenBufferRead[gScreenBufferReadPointer++];
andrewm@10 264
andrewm@10 265 // Rescale screen coordinates to matrix ranges; invert the Y
andrewm@10 266 // coordinate to go from normal screen coordinates to scope coordinates
andrewm@311 267 analogWriteOnce(context, n, gOutputX, constrain(map(x, 0, gScreenWidth, 0, 1.0), 0, 1.0));
andrewm@311 268 analogWriteOnce(context, n, gOutputY, constrain(map(y, 0, gScreenHeight, 1.0, 0), 0, 1.0));
andrewm@10 269 }
andrewm@10 270 else {
andrewm@10 271 // Still not ready! Write 0 until something happens
andrewm@311 272 analogWriteOnce(context, n, gOutputX, 0);
andrewm@311 273 analogWriteOnce(context, n, gOutputY, 0);
andrewm@10 274 }
andrewm@10 275
andrewm@10 276 if(gameStatusWinner() != 0) {
andrewm@10 277 // Blink one LED to show who won
andrewm@10 278 // Blink both LEDs when projectile is in motion
andrewm@22 279 float val = (gSampleCounter % 4000 > 2000) ? 1.0 : 0;
andrewm@311 280 analogWriteOnce(context, n, gOutputPlayer1LED, gameStatusWinner() == 1 ? val : 0);
andrewm@311 281 analogWriteOnce(context, n, gOutputPlayer2LED, gameStatusWinner() == 2 ? val : 0);
andrewm@10 282
andrewm@10 283 // After 5 seconds, restart the game
andrewm@10 284 gSamplesSinceFinish++;
andrewm@10 285 if(gSamplesSinceFinish > 22050*5)
andrewm@10 286 gGameShouldRestart = true;
andrewm@10 287 }
andrewm@10 288 else if(gameStatusProjectileInMotion()) {
andrewm@10 289 // Blink both LEDs when projectile is in motion
andrewm@22 290 float val = (gSampleCounter % 2000 > 1000) ? 1.0 : 0;
andrewm@311 291 analogWriteOnce(context, n, gOutputPlayer1LED, val);
andrewm@311 292 analogWriteOnce(context, n, gOutputPlayer2LED, val);
andrewm@10 293 }
andrewm@10 294 else if(gameStatusPlayer1Turn()) {
andrewm@311 295 analogWriteOnce(context, n, gOutputPlayer1LED, 1.0);
andrewm@311 296 analogWriteOnce(context, n, gOutputPlayer2LED, 0);
andrewm@10 297 }
andrewm@10 298 else {
andrewm@311 299 analogWriteOnce(context, n, gOutputPlayer2LED, 1.0);
andrewm@311 300 analogWriteOnce(context, n, gOutputPlayer1LED, 0);
andrewm@10 301 }
andrewm@10 302
andrewm@10 303 // Check if we have reached the point where we should next update
andrewm@10 304 if(gScreenBufferReadPointer >= gScreenBufferNextUpdateLocation &&
andrewm@10 305 !gScreenNextBufferReady) {
andrewm@10 306 // Update the screen at lower priority than the audio thread
giuliomoro@301 307 Bela_scheduleAuxiliaryTask(gScreenUpdateTask);
andrewm@10 308 }
andrewm@10 309
andrewm@10 310 gSampleCounter++;
andrewm@10 311 }
andrewm@10 312 }
andrewm@10 313
andrewm@10 314 void screen_update()
andrewm@10 315 {
andrewm@10 316 // If we should restart, reinitialise the game
andrewm@10 317 if(gGameShouldRestart) {
andrewm@10 318 restartGame();
andrewm@10 319 gGameShouldRestart = false;
andrewm@10 320 gSamplesSinceFinish = 0;
andrewm@10 321 }
andrewm@10 322
andrewm@10 323 // Render the game based on the current state
andrewm@10 324 gScreenBufferWriteLength = drawGame(gScreenBufferWrite, gScreenBufferMaxLength);
andrewm@10 325
andrewm@10 326 // Flag it as ready to go
andrewm@10 327 gScreenNextBufferReady = true;
andrewm@10 328 }
andrewm@10 329
andrewm@56 330 // cleanup() is called once at the end, after the audio has stopped.
andrewm@56 331 // Release any resources that were allocated in setup().
andrewm@10 332
giuliomoro@301 333 void cleanup(BelaContext *context, void *userData)
andrewm@10 334 {
andrewm@10 335 // Clean up the game state
andrewm@10 336 cleanupGame();
andrewm@10 337 }