Self-restarting MathKernel - is it possible in Mathematica?

前端 未结 6 1235
慢半拍i
慢半拍i 2021-02-01 10:06

This question comes from the recent question \"Correct way to cap Mathematica memory use?\"

I wonder, is it possible to programmatically restart MathKernel keeping the c

相关标签:
6条回答
  • 2021-02-01 10:42

    I have a similar requirement when I run a CUDAFunction for a long loop and CUDALink ran out of memory (similar here: https://mathematica.stackexchange.com/questions/31412/cudalink-ran-out-of-available-memory). There's no improvement on the memory leak even with the latest Mathematica 10.4 version. I figure out a workaround here and hope that you may find it's useful. The idea is that you use a bash script to call a Mathematica program (run in batch mode) multiple times with passing parameters from the bash script. Here is the detail instruction and demo (This is for Window OS):

    • To use bash-script in Win_OS you need to install cygwin (https://cygwin.com/install.html).
    • Convert your mathematica notebook to package (.m) to be able to use in script mode. If you save your notebook using "Save as.." all the command will be converted to comments (this was noted by Wolfram Research), so it's better that you create a package (File->New-Package), then copy and paste your commands to that.
    • Write the bash script using Vi editor (instead of Notepad or gedit for window) to avoid the problem of "\r" (http://www.linuxquestions.org/questions/programming-9/shell-scripts-in-windows-cygwin-607659/).

    Here is a demo of the test.m file

    str=$CommandLine;
    len=Length[str];
    Do[
    If[str[[i]]=="-start",
    start=ToExpression[str[[i+1]]];
    Pause[start];
    Print["Done in ",start," second"];
    ];
    ,{i,2,len-1}];
    

    This mathematica code read the parameter from a commandline and use it for calculation. Here is the bash script (script.sh) to run test.m many times with different parameters.

    #c:\cygwin64\bin\bash
    for ((i=2;i<10;i+=2))
    do
    math -script test.m -start $i
    done
    

    In the cygwin terminal type "chmod a+x script.sh" to enable the script then you can run it by typing "./script.sh".

    0 讨论(0)
  • 2021-02-01 10:43

    You can programmatically terminate the kernel using Exit[]. The front end (notebook) will automatically start a new kernel when you next try to evaluate an expression.

    Preserving "some code from the previous kernel" is going to be more difficult. You have to decide what you want to preserve. If you think you want to preserve everything, then there's no point in restarting the kernel. If you know what definitions you want to save, you can use DumpSave to write them to a file before terminating the kernel, and then use << to load that file into the new kernel.

    On the other hand, if you know what definitions are taking up too much memory, you can use Unset, Clear, ClearAll, or Remove to remove those definitions. You can also set $HistoryLength to something smaller than Infinity (the default) if that's where your memory is going.

    0 讨论(0)
  • 2021-02-01 10:44

    Perhaps the parallel computation machinery could be used for this? Here is a crude set-up that illustrates the idea:

    Needs["SubKernels`LocalKernels`"]
    
    doSomeWork[input_] := {$KernelID, Length[input], RandomReal[]}
    
    getTheJobDone[] :=
      Module[{subkernel, initsub, resultSoFar = {}}
      , initsub[] :=
          ( subkernel = LaunchKernels[LocalMachine[1]]
          ; DistributeDefinitions["Global`"]
          )
      ; initsub[]
      ; While[Length[resultSoFar] < 1000
        , DistributeDefinitions[resultSoFar]
        ; Quiet[ParallelEvaluate[doSomeWork[resultSoFar], subkernel]] /.
            { $Failed :> (Print@"Ouch!"; initsub[])
            , r_ :> AppendTo[resultSoFar, r]
            }
        ]
      ; CloseKernels[subkernel]
      ; resultSoFar
      ]
    

    This is an over-elaborate setup to generate a list of 1,000 triples of numbers. getTheJobDone runs a loop that continues until the result list contains the desired number of elements. Each iteration of the loop is evaluated in a subkernel. If the subkernel evaluation fails, the subkernel is relaunched. Otherwise, its return value is added to the result list.

    To try this out, evaluate:

    getTheJobDone[]
    

    To demonstrate the recovery mechanism, open the Parallel Kernel Status window and kill the subkernel from time-to-time. getTheJobDone will feel the pain and print Ouch! whenever the subkernel dies. However, the overall job continues and the final result is returned.

    The error-handling here is very crude and would likely need to be bolstered in a real application. Also, I have not investigated whether really serious error conditions in the subkernels (like running out of memory) would have an adverse effect on the main kernel. If so, then perhaps subkernels could kill themselves if MemoryInUse[] exceeded a predetermined threshold.

    Update - Isolating the Main Kernel From Subkernel Crashes

    While playing around with this framework, I discovered that any use of shared variables between the main kernel and subkernel rendered Mathematica unstable should the subkernel crash. This includes the use of DistributeDefinitions[resultSoFar] as shown above, and also explicit shared variables using SetSharedVariable.

    To work around this problem, I transmitted the resultSoFar through a file. This eliminated the synchronization between the two kernels with the net result that the main kernel remained blissfully unaware of a subkernel crash. It also had the nice side-effect of retaining the intermediate results in the event of a main kernel crash as well. Of course, it also makes the subkernel calls quite a bit slower. But that might not be a problem if each call to the subkernel performs a significant amount of work.

    Here are the revised definitions:

    Needs["SubKernels`LocalKernels`"]
    
    doSomeWork[] := {$KernelID, Length[Get[$resultFile]], RandomReal[]}
    
    $resultFile = "/some/place/results.dat";
    
    getTheJobDone[] :=
      Module[{subkernel, initsub, resultSoFar = {}}
      , initsub[] :=
          ( subkernel = LaunchKernels[LocalMachine[1]]
          ; DistributeDefinitions["Global`"]
          )
      ; initsub[]
      ; While[Length[resultSoFar] < 1000
        , Put[resultSoFar, $resultFile]
        ; Quiet[ParallelEvaluate[doSomeWork[], subkernel]] /.
            { $Failed :> (Print@"Ouch!"; CloseKernels[subkernel]; initsub[])
            , r_ :> AppendTo[resultSoFar, r]
            }
        ]
      ; CloseKernels[subkernel]
      ; resultSoFar
      ]
    
    0 讨论(0)
  • 2021-02-01 10:52

    Sounds like a job for CleanSlate.

    << Utilities`CleanSlate`;
    CleanSlate[]
    

    From: http://library.wolfram.com/infocenter/TechNotes/4718/

    "CleanSlate, tries to do everything possible to return the kernel to the state it was in when the CleanSlate.m package was initially loaded."

    0 讨论(0)
  • 2021-02-01 10:57

    The following approach runs one kernel to open a front-end with its own kernel, which is then closed and reopened, renewing the second kernel.

    This file is the MathKernel input, C:\Temp\test4.m

    Needs["JLink`"];
    $FrontEndLaunchCommand="Mathematica.exe";
    UseFrontEnd[
    nb = NotebookOpen["C:\\Temp\\run.nb"];
    SelectionMove[nb, Next, Cell];
    SelectionEvaluate[nb];
    ];
    Pause[8];
    CloseFrontEnd[];
    Pause[1];
    UseFrontEnd[
    nb = NotebookOpen["C:\\Temp\\run.nb"];
    Do[SelectionMove[nb, Next, Cell],{12}];
    SelectionEvaluate[nb];
    ];
    Pause[8];
    CloseFrontEnd[];
    Print["Completed"]
    

    The demo notebook, C:\Temp\run.nb contains two cells:

    x1 = 0;
    Module[{}, 
     While[x1 < 1000000, 
      If[Mod[x1, 100000] == 0, Print["x1=" <> ToString[x1]]]; x1++];
     NotebookSave[EvaluationNotebook[]];
     NotebookClose[EvaluationNotebook[]]]
    
    Print[x1]
    x1 = 0;
    Module[{}, 
     While[x1 < 1000000, 
      If[Mod[x1, 100000] == 0, Print["x1=" <> ToString[x1]]]; x1++];
     NotebookSave[EvaluationNotebook[]];
     NotebookClose[EvaluationNotebook[]]]
    

    The initial kernel opens a front-end and runs the first cell, then it quits the front-end, reopens it and runs the second cell.

    The whole thing can be run either by pasting (in one go) the MathKernel input into a kernel session, or it can be run from a batch file, e.g. C:\Temp\RunTest2.bat

    @echo off
    setlocal
    PATH = C:\Program Files\Wolfram Research\Mathematica\8.0\;%PATH%
    echo Launching MathKernel %TIME%
    start MathKernel -noprompt -initfile "C:\Temp\test4.m"
    ping localhost -n 30 > nul
    echo Terminating MathKernel %TIME%
    taskkill /F /FI "IMAGENAME eq MathKernel.exe" > nul
    endlocal
    

    It's a little elaborate to set up, and in its current form it depends on knowing how long to wait before closing and restarting the second kernel.

    0 讨论(0)
  • 2021-02-01 11:04

    From a comment by Arnoud Buzing yesterday, on Stack Exchange Mathematica chat, quoting entirely:

    In a notebook, if you have multiple cells you can put Quit in a cell by itself and set this option:

    SetOptions[$FrontEnd, "ClearEvaluationQueueOnKernelQuit" -> False]
    

    Then if you have a cell above it and below it and select all three and evaluate, the kernel will Quit but the frontend evaluation queue will continue (and restart the kernel for the last cell).

    -- Arnoud Buzing

    0 讨论(0)
提交回复
热议问题