Skip to content

Commit 215b037

Browse files
committed
FreeJ2ME: Introduce support for audio stream dumping
Mostly for debugging purposes, but it's also neat to have the ability to dump audio data from J2ME apps, especially ones that are encrypted and can't be easily opened up. Supported on both AWT and Libretro. As for Anbu/SDL... that one is in need of a UI rewrite, as it's still too barebones to have features be tacked onto it. The dump file path is (related to the jar): FreeJ2MEDumps/Audio/appname/*
1 parent a9235ca commit 215b037

File tree

6 files changed

+103
-15
lines changed

6 files changed

+103
-15
lines changed

src/javax/microedition/media/Manager.java

+40
Original file line numberDiff line numberDiff line change
@@ -16,19 +16,59 @@
1616
*/
1717
package javax.microedition.media;
1818

19+
import java.io.ByteArrayInputStream;
20+
import java.io.ByteArrayOutputStream;
1921
import java.io.InputStream;
22+
import java.io.File;
23+
import java.io.FileOutputStream;
2024
import java.io.IOException;
25+
import java.io.OutputStream;
2126

27+
import org.recompile.mobile.Mobile;
2228
import org.recompile.mobile.PlatformPlayer;
2329

2430
public final class Manager
2531
{
2632
public static final String TONE_DEVICE_LOCATOR = "device://tone";
2733
public static Player midiPlayers[] = new Player[32]; /* Default max amount of players in FreeJ2ME's config */
2834
public static byte midiPlayersIndex = 0;
35+
public static boolean dumpAudioStreams = false;
36+
public static short audioDumpIndex = 0;
2937

3038
public static Player createPlayer(InputStream stream, String type) throws IOException, MediaException
3139
{
40+
if(dumpAudioStreams)
41+
{
42+
// Copy the stream contents into a temporary stream to be saved as file
43+
final ByteArrayOutputStream streamCopy = new ByteArrayOutputStream();
44+
final byte[] copyBuffer = new byte[1024];
45+
int copyLength;
46+
while ((copyLength = stream.read(copyBuffer)) > -1 ) { streamCopy.write(copyBuffer, 0, copyLength); }
47+
streamCopy.flush();
48+
49+
// Make sure the initial strem will still be available for FreeJ2ME
50+
stream = new ByteArrayInputStream(streamCopy.toByteArray());
51+
52+
// And save the copy to the specified dir
53+
54+
OutputStream outStream;
55+
String dumpPath = "." + File.separatorChar + "FreeJ2MEDumps" + File.separatorChar + "Audio" + File.separatorChar + Mobile.getPlatform().loader.suitename + File.separatorChar;
56+
File dumpFile = new File(dumpPath);
57+
58+
if (!dumpFile.isDirectory()) { dumpFile.mkdirs(); }
59+
60+
if(type.equalsIgnoreCase("audio/mid") || type.equalsIgnoreCase("audio/midi") || type.equalsIgnoreCase("sp-midi") || type.equalsIgnoreCase("audio/spmidi"))
61+
{ dumpFile = new File(dumpPath + "Stream" + Short.toString(audioDumpIndex) + ".mid");}
62+
else if(type.equalsIgnoreCase("audio/x-wav") || type.equalsIgnoreCase("audio/wav")) { dumpFile = new File(dumpPath + "Stream" + Short.toString(audioDumpIndex) + ".wav");}
63+
else if(type.equalsIgnoreCase("audio/mpeg") || type.equalsIgnoreCase("audio/mp3")) { dumpFile = new File(dumpPath + "Stream" + Short.toString(audioDumpIndex) + ".mp3");}
64+
65+
outStream = new FileOutputStream(dumpFile);
66+
67+
streamCopy.writeTo(outStream);
68+
69+
audioDumpIndex++;
70+
}
71+
3272
//System.out.println("Create Player Stream "+type);
3373
if(type.equalsIgnoreCase("audio/mid") || type.equalsIgnoreCase("audio/midi") || type.equalsIgnoreCase("sp-midi") || type.equalsIgnoreCase("audio/spmidi"))
3474
{

src/libretro/freej2me_libretro.c

+16-7
Original file line numberDiff line numberDiff line change
@@ -146,6 +146,7 @@ int gameFPS; /* Auto(0), 60, 30, 15 */
146146
int soundEnabled; /* also acts as a boolean */
147147
int customMidi; /* Also acts as a boolean */
148148
int maxMidiPlayers; /* Maximum amount of MIDI Players allowed on FreeJ2ME at any given time */
149+
int dumpAudioStreams;
149150
/* Variables used to manage the pointer speed when controlled from an analog stick */
150151
int pointerXSpeed = 8;
151152
int pointerYSpeed = 8;
@@ -356,6 +357,13 @@ static void check_variables(bool first_time_startup)
356357
else if (!strcmp(var.value, "96")) { maxMidiPlayers = 96; }
357358
}
358359

360+
var.key = "freej2me_dumpaudiostreams";
361+
if (Environ(RETRO_ENVIRONMENT_GET_VARIABLE, &var) && var.value)
362+
{
363+
if (!strcmp(var.value, "off")) { dumpAudioStreams = 0; }
364+
else if (!strcmp(var.value, "on")) { dumpAudioStreams = 1; }
365+
}
366+
359367
var.key = "freej2me_pointertype";
360368
if (Environ(RETRO_ENVIRONMENT_GET_VARIABLE, &var) && var.value)
361369
{
@@ -434,7 +442,7 @@ static void check_variables(bool first_time_startup)
434442

435443

436444
/* Prepare a string to pass those core options to the Java app */
437-
snprintf(options_update, PIPE_MAX_LEN, "FJ2ME_LR_OPTS:|%lux%lu|%d|%d|%d|%d|%d|%d|%d", screenRes[0], screenRes[1], rotateScreen, phoneType, gameFPS, soundEnabled, customMidi, maxMidiPlayers);
445+
snprintf(options_update, PIPE_MAX_LEN, "FJ2ME_LR_OPTS:|%lux%lu|%d|%d|%d|%d|%d|%d|%d", screenRes[0], screenRes[1], rotateScreen, phoneType, gameFPS, soundEnabled, customMidi, maxMidiPlayers, dumpAudioStreams);
438446
optstrlen = strlen(options_update);
439447

440448
/* 0xD = 13, which is the special case where the java app will receive the updated configs */
@@ -494,7 +502,7 @@ void retro_init(void)
494502
*/
495503
check_variables(true);
496504

497-
char resArg[2][4], rotateArg[2], phoneArg[2], fpsArg[3], soundArg[2], midiArg[2], maxMidiArg[3];
505+
char resArg[2][4], rotateArg[2], phoneArg[2], fpsArg[3], soundArg[2], midiArg[2], maxMidiArg[3], dumpAudioArg[2];
498506
sprintf(resArg[0], "%lu", screenRes[0]); /* Libretro config Width */
499507
sprintf(resArg[1], "%lu", screenRes[1]); /* Libretro config Height */
500508
sprintf(rotateArg, "%d", rotateScreen);
@@ -503,16 +511,17 @@ void retro_init(void)
503511
sprintf(soundArg, "%d", soundEnabled);
504512
sprintf(midiArg, "%d", customMidi);
505513
sprintf(maxMidiArg,"%d", maxMidiPlayers);
514+
sprintf(dumpAudioArg, "%d", dumpAudioStreams);
506515

507516
/* start java process */
508517
char *javapath;
509518
Environ(RETRO_ENVIRONMENT_GET_SYSTEM_DIRECTORY, &javapath);
510519
char *outPath = malloc(sizeof(char) * PATH_MAX_LENGTH);
511520
fill_pathname_join(outPath, javapath, "freej2me-lr.jar", PATH_MAX_LENGTH);
512-
char *params[] = { "java", "-jar", outPath, resArg[0], resArg[1], rotateArg, phoneArg, fpsArg, soundArg, midiArg, maxMidiArg, NULL };
521+
char *params[] = { "java", "-jar", outPath, resArg[0], resArg[1], rotateArg, phoneArg, fpsArg, soundArg, midiArg, maxMidiArg, dumpAudioArg, NULL };
513522

514523
log_fn(RETRO_LOG_INFO, "Passing params: %s | %s | %s | %s | %s | %s | %s | %s | %s\n", *(params+3),
515-
*(params+4), *(params+5), *(params+6), *(params+7), *(params+8), *(params+9), *(params+10));
524+
*(params+4), *(params+5), *(params+6), *(params+7), *(params+8), *(params+9), *(params+10), *(params+11));
516525
log_fn(RETRO_LOG_INFO, "Preparing to open FreeJ2ME's Java app (make sure freej2me-lr.jar is inside system/).\n");
517526

518527
#ifdef __linux__
@@ -1005,7 +1014,7 @@ pid_t javaOpen(char *cmd, char **params)
10051014

10061015
log_fn(RETRO_LOG_INFO, "Opening: %s %s %s ...\n", *(params+0), *(params+1), *(params+2));
10071016
log_fn(RETRO_LOG_INFO, "Params: %s | %s | %s | %s | %s | %s | %s | %s | %s\n", *(params+3),
1008-
*(params+4), *(params+5), *(params+6), *(params+7), *(params+8), *(params+9), *(params+10));
1017+
*(params+4), *(params+5), *(params+6), *(params+7), *(params+8), *(params+9), *(params+10), *(params+11));
10091018

10101019
int fd_stdin = 0;
10111020
int fd_stdout = 1;
@@ -1134,15 +1143,15 @@ void javaOpen(char *cmd, char **params)
11341143
sprintf(cmdWin, "javaw -jar %s", cmd);
11351144

11361145
log_fn(RETRO_LOG_INFO, "Opening: %s \n", cmd);
1137-
for (int i = 3; i <= 10; i++) /* There are 10 cmd arguments for now */
1146+
for (int i = 3; i <= 11; i++) /* There are 10 cmd arguments for now */
11381147
{
11391148
log_fn(RETRO_LOG_INFO, "Processing arg %d: %s \n", i, *(params+i));
11401149
sprintf(cmdWin, "%s %s", cmdWin, *(params+i));
11411150
}
11421151

11431152
log_fn(RETRO_LOG_INFO, "Creating proc: %s \n", cmdWin);
11441153
log_fn(RETRO_LOG_INFO, "Params: %s | %s | %s | %s | %s | %s | %s | %s | %s\n", *(params+3),
1145-
*(params+4), *(params+5), *(params+6), *(params+7), *(params+8), *(params+9), *(params+10));
1154+
*(params+4), *(params+5), *(params+6), *(params+7), *(params+8), *(params+9), *(params+10), *(params+11));
11461155

11471156
GetStartupInfo(&startInfo);
11481157
startInfo.dwFlags = STARTF_USESTDHANDLES;

src/libretro/freej2me_libretro.h

+35-6
Original file line numberDiff line numberDiff line change
@@ -195,6 +195,20 @@ struct retro_core_option_v2_definition core_options[] =
195195
},
196196
"32"
197197
},
198+
{
199+
"freej2me_dumpaudiostreams",
200+
"Advanced Settings > Dump Audio Streams",
201+
"Dump Audio Streams",
202+
"This option allows FreeJ2ME to dump incoming Audio Data into $SYSTEM/FreeJ2MEDumps/Audio/appname/*, mostly useful for debugging",
203+
"This option allows FreeJ2ME to dump incoming Audio Data into $SYSTEM/FreeJ2MEDumps/Audio/appname/*, mostly useful for debugging",
204+
"advanced_settings",
205+
{
206+
{ "off", "Disable" },
207+
{ "on", "Enable" },
208+
{ NULL, NULL },
209+
},
210+
"off"
211+
},
198212
{
199213
"freej2me_pointertype",
200214
"Advanced Settings > Pointer Type",
@@ -421,6 +435,17 @@ struct retro_core_option_definition core_options_v1 [] =
421435
},
422436
"32"
423437
},
438+
{
439+
"freej2me_dumpaudiostreams",
440+
"Dump Audio Streams",
441+
"This option allows FreeJ2ME to dump incoming Audio Data into $SYSTEM/FreeJ2MEDumps/Audio/appname/*, mostly useful for debugging",
442+
{
443+
{ "off", "Disable" },
444+
{ "on", "Enable" },
445+
{ NULL, NULL },
446+
},
447+
"off"
448+
},
424449
{
425450
"freej2me_pointertype",
426451
"Pointer Type",
@@ -547,29 +572,33 @@ static const struct retro_variable vars[] =
547572
"freej2me_maxmidiplayers",
548573
"Max MIDI Players: 32|1|2|4|8|16|48|64|96"
549574
},
575+
{ /* Dump Audio Streams */
576+
"freej2me_dumpaudiostreams",
577+
"Dump Audio Streams: off|on"
578+
},
550579
{ /* Pointer Type */
551580
"freej2me_pointertype",
552-
"Pointer Type; Mouse|Touch|None",
581+
"Pointer Type; Mouse|Touch|None"
553582
},
554583
{ /* Screen Pointer X Speed */
555584
"freej2me_pointerxspeed",
556-
"Pointer X Speed; 4|1|2|8|16",
585+
"Pointer X Speed; 4|1|2|8|16"
557586
},
558587
{ /* Screen Pointer Y Speed */
559588
"freej2me_pointeryspeed",
560-
"Pointer Y Speed; 4|1|2|8|16",
589+
"Pointer Y Speed; 4|1|2|8|16"
561590
},
562591
{ /* Pointer's inner color */
563592
"freej2me_pointerinnercolor",
564-
"Pointer Inner Color; White|Red|Green|Blue|Yellow|Pink|Cyan|Black",
593+
"Pointer Inner Color; White|Red|Green|Blue|Yellow|Pink|Cyan|Black"
565594
},
566595
{ /* Pointer's outline color */
567596
"freej2me_pointeroutercolor",
568-
"Pointer Outline Color; Black|Red|Green|Blue|Yellow|Pink|Cyan|White",
597+
"Pointer Outline Color; Black|Red|Green|Blue|Yellow|Pink|Cyan|White"
569598
},
570599
{ /* Pointer's click indicator color */
571600
"freej2me_pointerclickcolor",
572-
"Pointer Click Indicator Color; Yellow|Black|Red|Green|Blue|Pink|Cyan|White",
601+
"Pointer Click Indicator Color; Yellow|Black|Red|Green|Blue|Pink|Cyan|White"
573602
},
574603
{ NULL, NULL },
575604
};

src/org/recompile/freej2me/AWTGUI.java

+4-2
Original file line numberDiff line numberDiff line change
@@ -23,6 +23,8 @@
2323

2424
import java.util.Arrays;
2525

26+
import javax.microedition.media.Manager;
27+
2628
public final class AWTGUI
2729
{
2830
/* This is used to indicate to FreeJ2ME that it has to call "settingsChanged()" to apply changes made here */
@@ -740,8 +742,8 @@ public void itemStateChanged(ItemEvent e)
740742
{
741743
public void itemStateChanged(ItemEvent e)
742744
{
743-
if(dumpAudioData.getState()){ /* TODO */ }
744-
else{ /* TODO */ }
745+
if(dumpAudioData.getState()){ Manager.dumpAudioStreams = true; }
746+
else{ Manager.dumpAudioStreams = false; }
745747
}
746748
});
747749

src/org/recompile/freej2me/Anbu.java

+2
Original file line numberDiff line numberDiff line change
@@ -71,6 +71,8 @@ public Anbu(String args[])
7171

7272
Mobile.setPlatform(new MobilePlatform(lcdWidth, lcdHeight));
7373

74+
/* TODO: Anbu has no way of enabling "Dump Audio Streams", a UI rewrite might be in order */
75+
7476
config = new Config();
7577
config.onChange = new Runnable() { public void run() { settingsChanged(); } };
7678

src/org/recompile/freej2me/Libretro.java

+6
Original file line numberDiff line numberDiff line change
@@ -135,6 +135,9 @@ public Libretro(String args[])
135135
maxmidistreams = Integer.parseInt(args[7]);
136136
Manager.updatePlayerNum((byte) maxmidistreams);
137137

138+
/* Dump Audio Streams will not be a per-game FreeJ2ME config, so it will have to be set every time for now */
139+
if(Integer.parseInt(args[8]) == 1) { Manager.dumpAudioStreams = true; }
140+
138141
/* Once it finishes parsing all arguments, it's time to set up freej2me-lr */
139142

140143
surface = new BufferedImage(lcdWidth, lcdHeight, BufferedImage.TYPE_INT_ARGB); // libretro display
@@ -376,6 +379,9 @@ public void run()
376379
if(Integer.parseInt(cfgtokens[8])==7) { config.settings.put("maxmidistreams", "64");}
377380
if(Integer.parseInt(cfgtokens[8])==8) { config.settings.put("maxmidistreams", "96");}
378381

382+
if(Integer.parseInt(cfgtokens[9])==1) { Manager.dumpAudioStreams = true; }
383+
if(Integer.parseInt(cfgtokens[9])==0) { Manager.dumpAudioStreams = false; }
384+
379385
config.saveConfig();
380386
settingsChanged();
381387
break;

0 commit comments

Comments
 (0)