Loading...
Logo
Processing Forum
I'm making an application that begins recording once it receives an audio signal above a threshold and ends recording once that falls below the threshold for a few seconds straight. I have set a maximum recording length to 5 minutes consecutively so that it doesn't run out of memory, yet after several recordings (last crash was 10 recordings x 5 minutes each) I get an OutOfMemoryError as it writes to the saved file.

Here is the error (minus the Processing added part):
Exception in thread "Animation Thread" java.lang.OutOfMemoryError: Java heap space
at org.tritonus.share.sampled.FloatSampleBuffer.convertToByteArray(FloatSampleBuffer.java:394)
at ddf.minim.javasound.JSBufferedSampleRecorder.save(JSBufferedSampleRecorder.java:144)
at ddf.minim.AudioRecorder.save(AudioRecorder.java:107)
at AOVAudioRecord.completeRecord(AOVAudioRecord.java:213)
at AOVAudioRecord.draw(AOVAudioRecord.java:136)
at processing.core.PApplet.handleDraw(PApplet.java:1606)
at processing.core.PApplet.run(PApplet.java:1503)
at java.lang.Thread.run(Thread.java:637)


two relevant functions:
Copy code
  1. void initRecord()
  2. {
  3.   currentFilename = getTimestamp() + ".wav";
  4.    recorder = minim.createRecorder(in, currentFilename, true);
  5.    worthSavingCounter.reset();
  6.    recorder.beginRecord();
  7. }

Copy code
  1. void completeRecord()
  2. {
  3.     recorder.endRecord();
  4.     recorder.save();
  5.     println("Done saving.");
  6.     addRecentFile(currentFilename);
  7.     currentRecordingLength = 0;
  8.     nRecordings++;
  9.     
  10. }

I can post all of the code if that helps. It seems clear its from saving the file, which is maxed out at 5 minutes in length, and only happens after it has recorded around 50 minutes worth... any suggestions for how I can fix this? I need this to run for 10 days straight

Replies(5)

I dramatically improved it just by setting the createRecorder()'s boolean buffered to false.


Currently the application has been running for about 15 hours, I've had constant audio input so it has recorded 230 wav files limited to 5 minutes each.

only problem is, the memory usage is still creeping upwards. 15 hours ago the application was using about 55mb, now its using 279.4mb. So with its current memory limit it would crash a little more than once every 3 days. Though this is not actually likely, because there isnt going to be constant audio to record like in my tests, it still is a vulnerability for my installation.

Any ideas how to fix this leak?
Hard to tell without seeing the source code of Minim...
As a wild guess, you can try and set recorder to null after save() call. It shouldn't be necessary, but who knows?

Note: you should try and get JVisualVM, a free tool (found in the JDK but also available separately), it can allow to track memory usage.
Hi I have a similar problem with a minim memory leak but in slight different configuration. I didn't know if it was the case to open a new topic, but anyhow: I'm using processing to control arduino where 2 photoresistors track the ammount of light and if it is over a certain ammount they will turn on their respective LED on the arduino board and play a random track from a pool of 10 different .aiff files.

I'm quite a noob and I might just be using the wrong option to select from a randomic pool of audio files in minim, but that's the only solution I came up with yesterday...obviously minim now does not seem to be able to track down the files in order to clear the memory.

Here's the code, any help? or any solution to achieve the same result in a more elegant way?

Copy code
  1. import ddf.minim.*;
  2. import ddf.minim.signals.*;
  3. import ddf.minim.analysis.*;
  4. import ddf.minim.effects.*;


  5. import processing.video.*;
  6. import processing.serial.*;


  7. import cc.arduino.*;

  8. Serial myPort;         // The Serial Port
  9. Arduino scheda;

  10. boolean ledA = false;
  11. int phoA = 0;

  12. boolean ledB = false;
  13. int phoB = 0;

  14. int brightness = 255;
  15. int fade = 1;

  16. float heightValA;
  17. float heightValB;

  18. int x = 0;
  19. int px= 0;
  20. int delayVal = 0;

  21. Minim audio = new Minim(this);
  22. AudioPlayer track;

  23. void setup(){
  24.   scheda = new Arduino(this, Arduino.list()[1], 57600);
  25.   size(800, 600);
  26.   rectMode(CENTER);
  27.   noStroke();
  28. }

  29. void draw() {
  30.   //set a random track to play
  31.   int randomTrack = (int)random(1, 11);
  32.   track = audio.loadFile("voce" + randomTrack + ".aiff");
  33.   
  34.   //read analog pin from arduino (photoresistors)
  35.   phoA = scheda.analogRead(0);
  36.   phoB = scheda.analogRead(1);
  37.   //map photoresistors val to fit window size
  38.   heightValA = map(phoA, 0,1023, 0, height);
  39.   heightValB = map(phoB, 0,1023, 0, height);
  40.   
  41.   //delay time: check every 1sec
  42.   if (millis() - delayVal > 1000) {
  43.     
  44.   if (phoA < 100 && ledB == false && phoB > 100) {
  45.     ledA = true;
  46.     track.play();
  47.     delayVal = millis();
  48.   } else if (phoB < 100) {
  49.     ledA = false;
  50.     ledB = true;
  51.     track.play();
  52.     delayVal = millis();
  53.   } else if (phoA < 100 && ledB == true) {
  54.     ledA = true;
  55.     ledB = false;
  56.     track.play();
  57.     delayVal = millis();
  58.   } else if (brightness == 0) {
  59.     ledA = false;
  60.     ledB = false;
  61.     brightness = 255;
  62.   }

  63.   }

  64.   //graphic placeholder to track photoresistors light values
  65.   noStroke();
  66.   fill(225, 10);
  67.   rect(width/2, height/2, width, height);
  68.   x = x+1;
  69.   if (x > width) {
  70.     x = 0;
  71.   }
  72.   stroke(250, 100, 0);
  73.   line(x, heightValA, px, heightValA);
  74.   stroke(0, 100,250);
  75.   line(x, heightValB, px, heightValB);
  76.   println(phoA);
  77.   ledState();
  78.   px = x;
  79. }


  80. void ledState() {
  81.  // LED state A 
  82.  if (ledA == true) {
  83.    scheda.digitalWrite(6, Arduino.HIGH);
  84.    noStroke();
  85.    fill(30, phoA);
  86.    rect(width/4,height/2, 10, 10);
  87. //     if (brightness > 0) {
  88. //       brightness -= fade;
  89. //     } else if (brightness <= 0) {
  90. //       brightness = 0;
  91. //     } 
  92.    
  93.  } else if (ledA == false) {
  94.    scheda.digitalWrite(6, Arduino.LOW); 
  95.  }
  96.  
  97.  // LED state B
  98.  if (ledB == true) {
  99.    scheda.digitalWrite(7, Arduino.HIGH);
  100.    noStroke();
  101.    fill(30, phoB); 
  102.    rect(width-width/4,height/2, 10, 10);
  103. //        if (brightness > 0) {
  104. //       brightness -= fade;
  105. //     } else if (brightness <= 0) {
  106. //       brightness = 0;
  107. //     } 

  108.  } else if (ledB == false) {
  109.    scheda.digitalWrite(7, Arduino.LOW); 
  110.  }
  111.  
  112. }

  113. //minim
  114. public void stop()
  115. {
  116.   track.close();
  117.   audio.stop();

  118. super.stop();
  119. }

and here follow the error message I get in processing, thanks in advance for any help you can give me!!


An OutOfMemoryError means that your code is either using up too much memory
because of a bug (e.g. creating an array that's too large, or unintentionally
loading thousands of images), or that your sketch may need more memory to run.
If your sketch uses a lot of memory (for instance if it loads a lot of data files)
you can increase the memory available to your sketch using the Preferences window.
Exception in thread "Animation Thread" java.lang.OutOfMemoryError: unable to create new native thread
at java.lang.Thread.start0(Native Method)
at java.lang.Thread.start(Thread.java:658)
at ddf.minim.javasound.JSBaseAudioRecordingStream.open(JSBaseAudioRecordingStream.java:399)
at ddf.minim.AudioSource.<init>(AudioSource.java:90)
at ddf.minim.AudioPlayer.<init>(AudioPlayer.java:57)
at ddf.minim.Minim.loadFile(Minim.java:360)
at ddf.minim.Minim.loadFile(Minim.java:341)
at tesiProj_audioLight.draw(tesiProj_audioLight.java:71)
at processing.core.PApplet.handleDraw(PApplet.java:1906)
at processing.core.PApplet.run(PApplet.java:1803)
at java.lang.Thread.run(Thread.java:680 


to make it easier this should be the "problematic part", where I assign the .aiff files every loop :/ and then I check if the conditions are "just right" so I call track.play() 


Copy code
  1. void draw() {
  2.   //set a random track to play
  3.   randomTrack = (int)random(1, 11);
  4.   track = audio.loadFile("voce" + randomTrack + ".aiff");
  5.   
  6.   //read analog pin from arduino (photoresistors)
  7.   phoA = scheda.analogRead(0);
  8.   phoB = scheda.analogRead(1);
  9.   //map photoresistors val to fit window size
  10.   heightValA = map(phoA, 0,1023, 0, height);
  11.   heightValB = map(phoB, 0,1023, 0, height);
  12.   
  13.   //delay time: check every 1sec
  14.   if (millis() - delayVal > 1000) {
  15.     
  16.   if (phoA < 100 && ledB == false && phoB > 100) {
  17.     ledA = true;
  18.     track.play();
  19.     delayVal = millis();
  20.   } else if (phoB < 100) {
  21.     ledA = false;
  22.     ledB = true;
  23.     track.play();
  24.     delayVal = millis();
  25.   } else if (phoA < 100 && ledB == true) {
  26.     ledA = true;
  27.     ledB = false;
  28.     track.play();
  29.     delayVal = millis();
  30.   } else if (brightness == 0) {
  31.     ledA = false;
  32.     ledB = false;
  33.     brightness = 255;
  34.   }

  35.   }

of course if i move
Copy code
  1.  track = audio.loadFile("voce" + randomTrack + ".aiff");
inside each "if" like so
Copy code
  1. if (millis() - delayVal > 1000) {
  2.     
  3.   if (phoA < 100 && ledB == false && phoB > 100) {
  4.     ledA = true;
  5.     track = audio.loadFile("voce" + randomTrack + ".aiff");
  6.     track.play();
  7.     delayVal = millis();
  8.   } else if (phoB < 100) {
  9.     ledA = false;
  10.     ledB = true;
  11.     track = audio.loadFile("voce" + randomTrack + ".aiff");
  12.     track.play();
  13.     delayVal = millis();
  14.   } else if (phoA < 100 && ledB == true) {
  15.     ledA = true;
  16.     ledB = false;
  17.     track = audio.loadFile("voce" + randomTrack + ".aiff");
  18.     track.play();
  19.     delayVal = millis();
  20.   } 
  21. }
it's not called 60 times per second but just 1 per second maximum; nonetheless I believe it'll lead to a memory leak just further in time...so it might crash again in 3hrs instead that in 30sec...any suggestion?

here's a screen taken from JVisualVM:



but I don't really know how to read it other than threads don't die!!
No luck I guess :/

Does anyone even have any suggestion on how to do this differently? like with an array or something and minim classes?

Thanks anyway :)