demo.diff 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412
  1. This mod adds demo (recorded gameplay) support to the SDL version of the game.
  2. This allows recording gameplay, making tool assisted speedruns by manually
  3. crafting game inputs or even saving the game at arbitrary positions. Rewinding
  4. of demos is supported. For details see the file demo.h and check the game help.
  5. Note that each demo is specific to game resolution, FPS and version! Also note
  6. that when using mouse, a long demo can get big (megabytes) as many mouse inputs
  7. have to be recorded (with keyboard only the demo will be relatively small).
  8. by drummyfish, released under CC0 1.0, public domain
  9. diff --git a/demo.h b/demo.h
  10. new file mode 100644
  11. index 0000000..94898be
  12. --- /dev/null
  13. +++ b/demo.h
  14. @@ -0,0 +1,275 @@
  15. +/**
  16. + @file demo.h
  17. +
  18. + Implementation of demo recording/playback for Anarch, can be used to record
  19. + gameplay or even make tool assisted runs.
  20. +
  21. + Demo file format: demo is a text file with one record per line. If line
  22. + starts with '#', it is ignored. First line of a demo should be a comment
  23. + holding the FPS, resolution (due to mouse offsets) and version of the game
  24. + used to record the demo. If a line starts with '*', it will be rewinded
  25. + to this position when played back. Otherwise the line is of format:
  26. +
  27. + F Z X Y
  28. +
  29. + where F is a non-negative frame number, X and Y are horizontal and vertical
  30. + mouse offsets (with possible sign), all in decimal, and Z is the button state
  31. + in format
  32. +
  33. + PONMLKJIHGFEDCBA
  34. +
  35. + where each letter is either 0 (not pressed) or 1 (pressed), A means key
  36. + with value 0 (SFG_KEY_UP), B means key with value 1 (SFG_KEY_RIGHT) etc.
  37. +
  38. + by drummyfish, released under CC0 1.0, public domain
  39. +*/
  40. +
  41. +#include <stdio.h>
  42. +#include <stdint.h>
  43. +#include "game.h"
  44. +
  45. +#define DEMO_FILENAME "demo.txt"
  46. +#define DEMO_MAXRECORDS 262144 ///< Maximum number of records in a demo.
  47. +
  48. +#define DEMO_NOTHING 0 ///< Do not use demo.
  49. +#define DEMO_REPLAY 1 ///< Load and replay demo.
  50. +#define DEMO_REPLAY_RECORD 2 ///< Load and replay demo, then record and save.
  51. +
  52. +#define DEMO_STATE_DONE 0
  53. +#define DEMO_STATE_REWINDING 1
  54. +#define DEMO_STATE_PLAYING 2
  55. +#define DEMO_STATE_RECORDING 3
  56. +
  57. +#define DEMO_PRINT(s) puts("DEMO: " s)
  58. +
  59. +typedef struct
  60. +{
  61. + uint32_t frame;
  62. + uint16_t buttonStates;
  63. + int16_t mouseDx;
  64. + int16_t mouseDy;
  65. +} DemoRecord;
  66. +
  67. +struct
  68. +{
  69. + uint32_t recordCount;
  70. + int32_t currentRecord;
  71. + uint32_t rewindTo;
  72. + uint8_t action;
  73. + uint8_t state;
  74. + DemoRecord records[DEMO_MAXRECORDS];
  75. +} demo;
  76. +
  77. +void _demoRecordInputs(void)
  78. +{
  79. + uint16_t oldB = 0;
  80. + int16_t oldDx = 0, oldDy = 0;
  81. +
  82. + if (demo.recordCount > 0)
  83. + {
  84. + oldB = demo.records[demo.recordCount - 1].buttonStates;
  85. + oldDx = demo.records[demo.recordCount - 1].mouseDx;
  86. + oldDy = demo.records[demo.recordCount - 1].mouseDy;
  87. + }
  88. +
  89. + uint16_t newB = 0;
  90. + int16_t newDx = 0, newDy = 0;
  91. +
  92. + SFG_getMouseOffset(&newDx,&newDy);
  93. +
  94. + for (uint8_t i = 0; i < SFG_KEY_COUNT; ++i)
  95. + {
  96. + newB <<= 1;
  97. + newB |= (SFG_keyPressed(SFG_KEY_COUNT - i - 1) != 0);
  98. + }
  99. +
  100. + if (newB != oldB || newDx != oldDx || newDy != oldDy)
  101. + {
  102. + if (demo.recordCount >= DEMO_MAXRECORDS - 1)
  103. + {
  104. + DEMO_PRINT("max records reached, stopping recording");
  105. + demo.state = DEMO_STATE_DONE;
  106. + }
  107. + else
  108. + {
  109. + demo.records[demo.recordCount].buttonStates = newB;
  110. + demo.records[demo.recordCount].mouseDx = newDx;
  111. + demo.records[demo.recordCount].mouseDy = newDy;
  112. + demo.records[demo.recordCount].frame = SFG_game.frame;
  113. + demo.recordCount++;
  114. + }
  115. + }
  116. +}
  117. +
  118. +// Call before executing a game (NOT rendering) frame. Returns the demo state.
  119. +uint8_t demoFrameStart()
  120. +{
  121. + switch (demo.state)
  122. + {
  123. + case DEMO_STATE_REWINDING:
  124. + if (SFG_game.frame >= demo.rewindTo)
  125. + {
  126. + DEMO_PRINT("rewinding finished, replaying");
  127. + demo.state = DEMO_STATE_PLAYING;
  128. + }
  129. +
  130. + case DEMO_STATE_PLAYING:
  131. + if (demo.currentRecord >= ((int32_t) demo.recordCount) - 1)
  132. + {
  133. + DEMO_PRINT("replaying done");
  134. + demo.state = DEMO_STATE_DONE;
  135. +
  136. + if (demo.action == DEMO_REPLAY_RECORD)
  137. + {
  138. + DEMO_PRINT("recording");
  139. + demo.state = DEMO_STATE_RECORDING;
  140. + }
  141. + }
  142. +
  143. + if (demo.state != DEMO_STATE_DONE &&
  144. + SFG_game.frame == demo.records[demo.currentRecord + 1].frame)
  145. + demo.currentRecord++;
  146. +
  147. + break;
  148. +
  149. + case DEMO_STATE_RECORDING:
  150. + _demoRecordInputs();
  151. + break;
  152. +
  153. + default: break;
  154. + }
  155. +
  156. + return demo.state;
  157. +}
  158. +
  159. +// Behaves the same as SFG_keyPressed but acts with the replayed demo.
  160. +int8_t demoKeyPressed(uint8_t key)
  161. +{
  162. + uint16_t b = demo.currentRecord >= 0 ?
  163. + demo.records[demo.currentRecord].buttonStates : 0;
  164. +
  165. + return (b >> key) & 0x01;
  166. +}
  167. +
  168. +// Behaves the same as SFG_getMouseOffset but acts with the replayed demo.
  169. +void demoGetMouseOffset(int16_t *x, int16_t *y)
  170. +{
  171. + *x = 0;
  172. + *y = 0;
  173. +
  174. + if (demo.currentRecord >= 0)
  175. + {
  176. + *x = demo.records[demo.currentRecord].mouseDx;
  177. + *y = demo.records[demo.currentRecord].mouseDy;
  178. + }
  179. +}
  180. +
  181. +/* Loads demo from demo file or just initializes a new demo. The action
  182. + parameter is DEMO_NOTHING, DEMO_REPLAY or DEMO_REPLAY_RECORD. */
  183. +void demoInit(uint8_t action)
  184. +{
  185. + DEMO_PRINT("initializing");
  186. + demo.action = action;
  187. +
  188. + if (action == DEMO_NOTHING)
  189. + {
  190. + demo.state = DEMO_STATE_DONE;
  191. + return;
  192. + }
  193. +
  194. + DEMO_PRINT("rewinding");
  195. +
  196. + demo.state = DEMO_STATE_REWINDING;
  197. + demo.currentRecord = -1;
  198. + demo.recordCount = 0;
  199. + demo.rewindTo = 0;
  200. +
  201. + char line[128];
  202. + FILE *file = fopen(DEMO_FILENAME,"r");
  203. +
  204. + if (file == NULL)
  205. + {
  206. + DEMO_PRINT("couldn't open demo file for reading");
  207. + return;
  208. + }
  209. +
  210. + while (fgets(line,128,file))
  211. + {
  212. + if (line[0] == '*')
  213. + {
  214. + if (demo.recordCount >= 1)
  215. + demo.rewindTo = demo.records[demo.recordCount - 1].frame + 1;
  216. + }
  217. + else if (line[0] != '#')
  218. + {
  219. + unsigned long f, b;
  220. + int dx, dy;
  221. +
  222. + if (demo.recordCount >= DEMO_MAXRECORDS)
  223. + {
  224. + DEMO_PRINT("demo too big");
  225. + return;
  226. + }
  227. +
  228. + if (sscanf(line," %lu %lu %d %d",&f,&b,&dx,&dy) != 4)
  229. + DEMO_PRINT("bad format of line in demo");
  230. +
  231. + if (demo.recordCount >= 1 &&
  232. + f <= demo.records[demo.recordCount - 1].frame)
  233. + {
  234. + DEMO_PRINT("demo has backwards records");
  235. + return;
  236. + }
  237. +
  238. + demo.records[demo.recordCount].buttonStates = 0;
  239. +
  240. + for (int i = 0; i < 32; ++i)
  241. + {
  242. + demo.records[demo.recordCount].buttonStates |= ((b % 10) >= 1) << i;
  243. + b /= 10;
  244. + }
  245. +
  246. + demo.records[demo.recordCount].frame = f;
  247. + demo.records[demo.recordCount].mouseDx = dx;
  248. + demo.records[demo.recordCount].mouseDy = dy;
  249. +
  250. + demo.recordCount++;
  251. + }
  252. + }
  253. +
  254. + fclose(file);
  255. +}
  256. +
  257. +// Call before program end.
  258. +void demoEnd(void)
  259. +{
  260. + if (demo.action == DEMO_REPLAY_RECORD)
  261. + {
  262. + DEMO_PRINT("saving to file");
  263. +
  264. + FILE *file = fopen(DEMO_FILENAME,"w");
  265. +
  266. + if (file == NULL)
  267. + {
  268. + DEMO_PRINT("couldn't open demo file for writing");
  269. + return;
  270. + }
  271. +
  272. + fprintf(file,"# Anarch demo, %d x %d, %d FPS, v. " SFG_VERSION_STRING "\n",
  273. + SFG_SCREEN_RESOLUTION_X,SFG_SCREEN_RESOLUTION_Y,SFG_FPS);
  274. +
  275. + for (int i = 0; i < demo.recordCount; ++i)
  276. + {
  277. + unsigned long b = 0;
  278. +
  279. + for (int j = 0; j < 32; ++j)
  280. + b = b * 10 +
  281. + ((demo.records[i].buttonStates & (((uint32_t) 0x01) << (31 - j))) != 0);
  282. +
  283. + fprintf(file,"%u %016lu %d %d\n",
  284. + demo.records[i].frame,b,demo.records[i].mouseDx,demo.records[i].mouseDy);
  285. + }
  286. +
  287. + fclose(file);
  288. + }
  289. +}
  290. diff --git a/main_sdl.c b/main_sdl.c
  291. index b2b28ae..9dd2649 100644
  292. --- a/main_sdl.c
  293. +++ b/main_sdl.c
  294. @@ -97,8 +97,13 @@
  295. #include <unistd.h>
  296. #include <SDL2/SDL.h>
  297. +void stepDemo(void);
  298. +
  299. +#define SFG_GAME_STEP_COMMAND stepDemo();
  300. +
  301. #include "game.h"
  302. #include "sounds.h"
  303. +#include "demo.h"
  304. const uint8_t *sdlKeyboardState;
  305. uint8_t webKeyboardState[SFG_KEY_COUNT];
  306. @@ -113,6 +118,11 @@ SDL_Renderer *renderer;
  307. SDL_Texture *texture;
  308. SDL_Surface *screenSurface;
  309. +void stepDemo(void)
  310. +{
  311. + demoFrameStart();
  312. +}
  313. +
  314. // now implement the Anarch API functions (SFG_*)
  315. void SFG_setPixel(uint16_t x, uint16_t y, uint8_t colorIndex)
  316. @@ -122,7 +132,7 @@ void SFG_setPixel(uint16_t x, uint16_t y, uint8_t colorIndex)
  317. uint32_t SFG_getTimeMs()
  318. {
  319. - return SDL_GetTicks();
  320. + return SDL_GetTicks() + demo.rewindTo * SFG_MS_PER_FRAME;
  321. }
  322. void SFG_save(uint8_t data[SFG_SAVE_SIZE])
  323. @@ -185,6 +195,12 @@ int8_t mouseMoved = 0; /* Whether the mouse has moved since program started,
  324. void SFG_getMouseOffset(int16_t *x, int16_t *y)
  325. {
  326. + if (demo.state == DEMO_STATE_REWINDING || demo.state == DEMO_STATE_PLAYING)
  327. + {
  328. + demoGetMouseOffset(x,y);
  329. + return;
  330. + }
  331. +
  332. #ifndef __EMSCRIPTEN__
  333. if (mouseMoved)
  334. {
  335. @@ -220,6 +236,9 @@ void SFG_processEvent(uint8_t event, uint8_t data)
  336. int8_t SFG_keyPressed(uint8_t key)
  337. {
  338. + if (demo.state == DEMO_STATE_REWINDING || demo.state == DEMO_STATE_PLAYING)
  339. + return demoKeyPressed(key);
  340. +
  341. if (webKeyboardState[key]) // this only takes effect in the web version
  342. return 1;
  343. @@ -368,6 +387,9 @@ void SFG_setMusic(uint8_t value)
  344. void SFG_playSound(uint8_t soundIndex, uint8_t volume)
  345. {
  346. + if (demo.state == DEMO_STATE_REWINDING)
  347. + return;
  348. +
  349. uint16_t pos = (audioPos +
  350. ((SFG_game.frame - audioUpdateFrame) * SFG_MS_PER_FRAME * 8)) %
  351. SFG_SFX_SAMPLE_COUNT;
  352. @@ -393,6 +415,7 @@ int main(int argc, char *argv[])
  353. uint8_t argHelp = 0;
  354. uint8_t argForceWindow = 0;
  355. uint8_t argForceFullscreen = 0;
  356. + uint8_t argDemo = DEMO_NOTHING;
  357. #ifndef __EMSCRIPTEN__
  358. argForceFullscreen = 1;
  359. @@ -409,6 +432,10 @@ int main(int argc, char *argv[])
  360. argForceWindow = 1;
  361. else if (argv[i][0] == '-' && argv[i][1] == 'f' && argv[i][2] == 0)
  362. argForceFullscreen = 1;
  363. + else if (argv[i][0] == '-' && argv[i][1] == 'd' && argv[i][2] == 0)
  364. + argDemo = DEMO_REPLAY;
  365. + else if (argv[i][0] == '-' && argv[i][1] == 'D' && argv[i][2] == 0)
  366. + argDemo = DEMO_REPLAY_RECORD;
  367. else
  368. puts("SDL: unknown argument");
  369. }
  370. @@ -423,7 +450,9 @@ int main(int argc, char *argv[])
  371. puts("CLI flags:\n");
  372. puts("-h print this help and exit");
  373. puts("-w force window");
  374. - puts("-f force fullscreen\n");
  375. + puts("-f force fullscreen");
  376. + puts("-d play demo file " DEMO_FILENAME);
  377. + puts("-D play and record (append) demo file " DEMO_FILENAME "\n");
  378. puts("controls:\n");
  379. puts("- arrows, numpad, [W] [S] [A] [D] [Q] [E]: movement");
  380. puts("- mouse: rotation, [LMB] shoot, [RMB] toggle free look");
  381. @@ -439,6 +468,8 @@ int main(int argc, char *argv[])
  382. return 0;
  383. }
  384. + demoInit(argDemo);
  385. +
  386. SFG_init();
  387. puts("SDL: initializing SDL");
  388. @@ -516,6 +547,8 @@ int main(int argc, char *argv[])
  389. mainLoopIteration();
  390. #endif
  391. + demoEnd();
  392. +
  393. puts("SDL: freeing SDL");
  394. SDL_GameControllerClose(sdlController);