How to efficiently pass large constant matrix in parfor
12 views (last 30 days)
Show older comments
Hi all,
I am trying to perform a task using parfor. However, it's inefficient since I have to load a large constant structure from a mat file every time I call the function. Is there a way I can load this structure in shared memory or use it as global variable or some thing else?
Here's a short description of simplified form of my problem.
- Let's say we have a structure "S". The size of this structure is ~15GB.
- This structure "S" is a constant every time we call the function "fun".
- All workers need "S".
Our Aim is as follows:
parfor j =1:1000
output = fun (S, j)
end
Now, we have four options to perform the above aim efficiently viz:
- Declare "S" as global variable. But as per my understanding, this is not possible in parfor.
- Have "S" in a shared memory so that all threads can access it since "S" is a constant for all the threads. But how do we do it?
- Make multiple copies of "S", and have each worked access a separate copy. But this may not be possible since size of "S" is very large and we may be bounded by available memory.
- Every time we call the function "fun", the function "fun" first loads the structure "S" from mat file and then works on it. However, this is very inefficient process.
Any thoughts?
Thanks in advance.
0 Comments
Answers (6)
Sean de Wolski
on 7 Mar 2014
One possibility:
Though this will require S to live on each worker concurrently (i.e. case 3 above). Do all workers need all of S or is there a way that you could break it into chunks and have each worker work on a chunk?
Walter Roberson
on 9 Jun 2018
Sean mentioned http://www.mathworks.com/matlabcentral/fileexchange/31972-worker-object-wrapper which was the appropriate supported way at the time he posted.
In R2015b, parallel.pool.constant() was added to permit a constant to be distributed to all of the workers.
The supported mechanism for global variables is to create the pool and then use parfevalOnAll to run code that initializes the global variable on the worker, using data copied in somehow (this does not use parallel.pool.constant() so data might be copied individually.)
Also supported is to use memmapfile() on each worker; however, this can effectively require that each worker read in the contents of the file.
Not supported but efficient is to use FEX:sharedmatrix which uses shared memory segments. Note that this will only work within any one node or at best across nodes that support unified memory such as NUMA; if you are working with clusters that have independent address spaces per node then you might need to distribute the memory to each node once, to be shared among the workers there.
2 Comments
Pavel Sinha
on 11 Jun 2018
Hi,
Thanks for the answer.
Now, if I have a matlab worker running some functions on GPU, can the functions running on GPU access global variables. Since this GPU function is no longer being executed on the Matlab worker, rather on the GPU, I was hoping, the global variables will be accessible from GPU.
Walter Roberson
on 12 Jun 2018
Unfortunately there are conflicts between the newest MATLAB and the newest CUDA on my particular operating system, so I cannot test this out.
I think a gpuArray object at the MATLAB level is a handle object that refers into the GPU, so I guess that storing one as a global variable and changing it within the same worker would be no different than passing the gpuArray object between different routines that might modify it.
But, to be clear, no matter what you do, the global variables from one worker will not be the same as the global variables from another worker. You cannot pass references to variables stored in the GPU between workers and expect to be able to access the same GPU memory from any of those workers.
Marta Salas
on 8 Mar 2014
Edited: Marta Salas
on 8 Mar 2014
You can use global variables on parfor but, you cannot define global variables or persistent variables within the body of the parfor loop. In your case, S is a constant you can define outside the parfor.
global S;
parfor j =1:1000
output = fun (j)
end
function fun(j)
global S
%your code
end
0 Comments
Pavel Sinha
on 9 Jun 2018
Edited: Walter Roberson
on 9 Jun 2018
Hi Marta,
In my code the global variable S, in the example above, which is being set before the parfor loop is being read as empty by the worker
so, if the do the following:
function fun(j)
global S
S
%your code
end
I get,
S =
[ ]
There must be a way for the workers to read and write to global variables. I fully understand that they variables will not be in sync with respect to other workers. I am fully aware of it. My problem right now is that I am simply not able to write or read global variables from with in a function that has been called inside a parfor loop.
Thanks,
Pavel
1 Comment
Walter Roberson
on 9 Jun 2018
The contents of global variables are never sent from the client to the workers, and are never copied back.
Pavel Sinha
on 9 Jun 2018
Edited: Walter Roberson
on 9 Jun 2018
clc;
close all;
clear all;
NO_PAR_POOLS = 3;
if isempty(gcp('nocreate'))
parpool(NO_PAR_POOLS);
end
global S;
S=10;
parfor j =1:3
if j==1; fun1(); end
if j==2; fun2(); end
if j==3; fun3(); end
end
function fun1()
global S
S
end
function fun2()
global S
l=S+10;
l
end
function fun3()
global S
k=S*10;
k
end
0 Comments
See Also
Categories
Find more on GPU Computing in Help Center and File Exchange
Community Treasure Hunt
Find the treasures in MATLAB Central and discover how the community can help you!
Start Hunting!