Skip to content

andypotatohy/roast

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

ROAST: Realistic vOlumetric-Approach-based Simulator for Transcranial electric stimulation

  1. Getting started
  2. How to use roast
  3. How to use roast_target
  4. More notes on the capInfo.xlsx file
  5. Outputs of ROAST software
  6. Review of simulation data
  7. How to ask questions
  8. Acknowledgements
  9. Notes
  10. License

1. Getting started

After you download the zip file, unzip it, launch your Matlab, make sure you are under the root directory of ROAST (i.e., you can see example/, lib/, and all other files), and then enter:

roast

This will demo a modeling process on the MNI152 head. Specifically, it will use the T1 image of the 6th gen MNI-152 head to build a TES model with anode on Fp1 (1 mA) and cathode on P4 (-1 mA).

There are 3 main functions that you can call: roast(), roast_target() and reviewRes(), which will be covered in Section 2, Section 3, and Section 6, respectively.

2. How to use roast

2.1 Synopsis of roast

roast(subj,recipe,varargin)

subj: file name (can be either relative or absolute path) of the MRI of the subject that you want to run simulation on. The MRI can be either T1 or T2. If you have both T1 and T2, then put T2 file in the option 'T2' (see below options for details). If you do not have any MRI but just want to run ROAST for a general result, you can use the default subject the MNI152 averaged head (see Example 1) or the New York head (see Example 2).

recipe: how you want to ROAST the subject you specified above. Default recipe is anode on Fp1 (1 mA) and cathode on P4 (-1 mA). You can specify any recipe you want in the format of electrodeName-injectedCurrent pair (see Example 3). You can pick any electrode from the 10/20, 10/10, 10/05, BioSemi-256, or EGI HCGSN-256 EEG system (see the Microsoft Excel file capInfo.xlsx under the root directory of ROAST). The unit of the injected current is in milliampere (mA). Make sure they sum up to 0. You can also place electrodes at customized locations on the scalp. See Example 5 for details. You can also use a special recipe called "leadField", so that ROAST will automatically generate all the data that will allow you to call roast_target() later to perform targeted TES. See Example 25 and How to use roast_target.

varargin: Options for ROAST can be entered as Name-Value pairs in the 3rd argument (available from ROAST v2.0). The syntax follows the Matlab convention (see plot() for example).

If you do not want to read the detailed info on the options below, you can go to Example 24 for quick reference.

'capType' -- the EEG system that you want to pick any electrode from.
'1020' | '1010' (default) | '1005' | 'BioSemi' | 'EGI'
You can also use customized electrode locations you defined. Just provide the text file that contains the electrode coordinates. See below Example 5 for details.

'elecType' -- the shape of electrode.
'disc' (default) | 'pad' | 'ring'
Note you can specify different shapes to different electrodes, i.e., you can place different types of electrodes at the same time. See below Example 6 for details.

'elecSize' -- the size of electrode.
All sizes are in the unit of millimeter (mm). For disc electrodes, sizes follow the format of [radius height], and default size is [6mm 2mm]; for pad electrodes, sizes follow the format of [length width height], and default size is [50mm 30mm 3mm]; for ring electrodes, sizes follow the format of [innerRadius outterRadius height], and default size is [4mm 6mm 2mm].

If you're placing only one type of electrode (e.g., either disc, or pad, or ring), you can use a one-row vector to customize the size, see Example 7; if you want to control the size for each electrode separately (provided you're placing only one type of electrode), you need to specify the size for each electrode correspondingly in a N-row matrix, where N is the number of electrodes to be placed, see Example 8; if you're placing more than one type of electrodes and also want to customize the sizes, you need to put the size of each electrode in a 1-by-N cell (put [] for any electrode that you want to use the default size), where N is the number of electrodes to be placed, see Example 9.

'elecOri' -- the orientation of pad electrode (ONLY applies to pad electrodes).
'lr' (default) | 'ap' | 'si' | direction vector of the long axis
For pad electrodes, you can define their orientation by giving the direction of the long axis. You can simply use the three pre-defined keywords:

  • 'lr'--long axis going left (l) and right (r);
  • 'ap'--long axis pointing front (anterior) and back (posterior);
  • 'si'--long axis going up (superior) and down (inferior).
    For other orientations you can also specify the direction precisely by giving the direction vector of the long axis.

If you're placing pad electrodes only, use the pre-defined keywords (Example 10) or the direction vector of the long axis (Example 11) to customize the orientations; if you want to control the orientation for each pad electrode separately, you need to specify the orientation for each pad correspondingly using the pre-defined keywords in a 1-by-N cell (Example 12) or the direction vectors of the long axis in a N-by-3 matrix (Example 13), where N is the number of pad electrodes to be placed; if you're placing more than one type of electrodes and also want to customize the pad orientations, you need to put the orientations into a N-by-3 matrix (Example 14; or just a 1-by-3 vector or a single pre-defined keyword if same orientation for all the pads) where N is the number of pad electrodes, or into a 1-by-N cell (Example 15), where N is the number of all electrodes to be placed (put [] for non-pad electrodes).

'T2' -- use a T2-weighted MRI to help segmentation.
[] (default) | file path to the T2 MRI
If you have a T2 MRI aside of T1, you can put the T2 file in this option, see Example 16, note you should put the T1 and T2 files in the same directory. If you ONLY have a T2 MRI, put the T2 file in the first argument 'subj' when you call roast, just like what you would do when you only have a T1.

'multipriors' -- use MultiPriors (a deep convolutional neural network) for segmentation.
'on' | 'off' (default)
For heads with abnormal anatomies such as a lesion, SPM usually cannot output a correct segmentation that captures the lesions. You can use a pre-trained deep convolutional neural network known as the MultiPriors for modeling these abnormal anatomies. To use MultiPriors instead of SPM for segmentation, simply turn on this option. See Example 17. Note if you turn on 'multipriors', please do not provide any T2 image (just providing T1 is enough), as MultiPriors only works with T1 images.

'meshOptions' -- advanced options of ROAST, for controlling mesh parameters (see Example 18).
5 sub-options are available:

  • meshOpt.radbound: maximal surface element size, default 5;
  • meshOpt.angbound: mimimal angle of a surface triangle, default 30;
  • meshOpt.distbound: maximal distance between the center of the surface bounding circle and center of the element bounding sphere, default 0.3;
  • meshOpt.reratio: maximal radius-edge ratio, default 3;
  • meshOpt.maxvol: target maximal tetrahedral element volume, default 10.
    See iso2mesh documentation for more details on these options.

'simulationTag' -- a unique tag that identifies each simulation.
dateTime string (default) | user-provided string
This tag is used by ROAST for managing simulation data. ROAST can identify if a certain simulation has been already run. If yes, it will just load the results to save time. You can leave this option empty so that ROAST will just use the date and time as the unique tag for the simulation. Or you can provide your preferred tag for a specific simulation (Example 19), then you can find it more easily later. Also all the simulation history with options info for each simulation are saved in the log file (named as "subjName_roastLog"), parsed by the simulation tags.

'resampling' -- re-sample the input MRI to 1mm isotropic resolution.
'on' | 'off' (default)
Sometimes the input MRI has a resolution of not being 1 mm, e.g., 0.6 mm. While higher resolution can give more accurate models, the computation will be more expensive and thus slower. If you want a faster simulation, you can ask ROAST to resample the MRI into 1 mm isotropic resolution by turning on this option (Example 20). Also it is recommended to turn on this option if your input MRI has anisotropic resolution (e.g., 1 mm by 1.2 mm by 1.2 mm), as the electrode size will not be exact if the model is built from an MRI with anisotropic resolution. From ROAST v3.0, the data generated by ROAST can be loaded into the Soterix software HD-Explore and HD-Targets, if the input MRI has 1 mm resolution. Users can turn on 'resampling' option for non-1 mm resolution MRI if they want to load the results (only the lead field) in Soterix software later for better visualization. See Example 26.

'zeroPadding' -- extend the input MRI by some amount, to avoid complications when electrodes are placed by the image boundaries. Default is not padding any slices to the MRI. You can ask ROAST to pad N empty slices in all the six directions of the input MRI (left, right, front, back, up and down), where N is a positive integer. This is very useful when placing big electrodes on the locations close to image boundaries (Example 21). This is also useful for MRIs that are cut off at the nose. If you specify a zeropadding of, say, 60 slices, ROAST can automatically get the segmentation of the lower part of the head, see (Example 21).

'conductivities' -- advanced options of ROAST, the values are stored as a structure, with the following field names:

  • white (default 0.126 S/m);
  • gray (default 0.276 S/m);
  • csf (default 1.65 S/m);
  • bone (default 0.01 S/m);
  • skin (default 0.465 S/m);
  • air (default 2.5e-14 S/m);
  • gel (default 0.3 S/m);
  • electrode (default 5.9e7 S/m).
    You can use this option to customize the electrical conductivity for each tissue, each electrode, as well as the conducting medium under each electrode. You can even assign different conductivity values to different electrodes and their conducting media (e.g., 'gel'). See Example 22 and Example 23 for details.

2.2 Examples on roast

Example 1

roast

Default call of ROAST, will demo a modeling process on the MNI152 head. Specifically, this will use the MRI of the MNI152 head to build a model of transcranial electric stimulation (TES) with anode on Fp1 (1 mA) and cathode on P4 (-1 mA). Electrodes are modeled by default as small disc electrodes. See options below for details.

Example 2

roast('nyhead')

ROAST New York head. Again this will run a simulation with anode on Fp1 (1 mA) and cathode on P4 (-1 mA), but on the 0.5-mm resolution New York head. A decent machine of 50GB memory and above is recommended for running New York head. Again electrodes are modeled by default as small disc electrodes. See options below for details.

Example 3

roast('example/subject1.nii',{'F1',0.3,'P2',0.7,'C5',-0.6,'O2',-0.4})

Build the TES model on any subject with your own "recipe". Here we inject 0.3 mA at electrode F1, 0.7 mA at P2, and we ask 0.6 mA coming out of C5, and 0.4 mA flowing out of O2. You can define any stimulation montage you want in the 2nd argument, with electrodeName-injectedCurrent pair. Electrodes are modeled by default as small disc electrodes. You can pick any electrode from the 10/20, 10/10, 10/05, BioSemi-256, or EGI HCGSN-256 EEG system. You can find all the info on electrodes (names, locations, coordinates) in the Microsoft Excel file capInfo.xlsx under the root directory of ROAST. Note the unit of the injected current is milliampere (mA). Make sure they sum up to 0.

Example 4

roast('example/subject1.nii',{'G12',1,'J7',-1},'captype','biosemi')

Run simulation on subject1 with anode on G12 (1 mA) and cathode on J7 (-1 mA) from the extended BioSemi-256 system (see capInfo.xlsx under the root directory of ROAST).

Example 5

roast('example/subject1.nii',{'E12',0.25,'E7',-0.25,'Nk1',0.5,'Nk3',-0.5,'custom1',0.25,'custom3',-0.25},'captype','egi')

Run simulation on subject1 with recipe that includes: EGI electrodes E12 and E7; neck electrodes Nk1 and Nk3 (see capInfo.xlsx); and user-provided electrodes custom1 and custom3. You can use a free program called MRIcro to load the MRI first (note do NOT use MRIcron for this as MRIcron will not give you the true voxel coordinates) and click the locations on the scalp surface where you want to place the electrodes, record the voxel coordinates returned by MRIcro into a text file, and save the text file to the MRI data directory with name "subjName_customLocations" (e.g., here for subject1 it's saved as "subject1_customLocations"). ROAST will load the text file and place the electrodes you specified. You need to name each customized electrode in the text file starting with "custom" (e.g., for this example they're named as custom1, custom2, etc. You can of course do "custom_MyPreferredElectrodeName"). Note that you need to use the MRI that enters ROAST to click for the customized electrode locations (here in this example you need to load 'example/subject1.nii'), even if the MRI is not in RAS orientation, or you turn on the 'resampling' or use 'zeroPadding' option. Just record whatever you get from the original MRI that you used for running ROAST, and ROAST will take care of all the transforms of MRI data (re-orientation into RAS, resampling or zero-padding).

Example 6

roast([],{'Fp1',1,'FC4',1,'POz',-2},'electype',{'disc','pad','ring'})

Run simulation on the MNI152 averaged head with the specified recipe. A disc electrode will be placed at location Fp1, a pad electrode will be placed at FC4, and a ring electrode will be placed at POz. The sizes and orientations will be set as default.

Example 7

roast('nyhead',[],'electype','ring','elecsize',[7 10 3])

Run simulation on the New York head with default recipe. Ring electrodes will be placed at Fp1 and P4. The size of each ring is 7mm inner radius, 10mm outter radius and 3mm height.

Example 8

roast('nyhead',{'Fp1',1,'FC4',1,'POz',-2},'electype','ring','elecsize',[7 10 3;6 8 3;4 6 2])

Run simulation on the New York head with the specified recipe. Ring electrode placed at Fp1 will have size [7mm 10mm 3mm]; ring at FC4 will have size [6mm 8mm 3mm]; and ring at POz will have size [4mm 6mm 2mm].

Example 9

roast([],{'Fp1',1,'FC4',1,'POz',-2},'electype',{'disc','pad','ring'},'elecsize',{[8 2],[45 25 4],[5 8 2]})

Run simulation on the MNI152 averaged head with the specified recipe. A disc electrode will be placed at location Fp1 with size [8mm 2mm], a pad electrode will be placed at FC4 with size [45mm 25mm 4mm], and a ring electrode will be placed at POz with size [5mm 8mm 2mm].

Example 10

roast([],[],'electype','pad','elecori','ap')

Run simulation on the MNI152 averaged head with default recipe. Pad electrodes will be placed at Fp1 and P4, with default size of [50mm 30mm 3mm] and the long axis will be oriented in the direction of front to back.

Example 11

roast([],[],'electype','pad','elecori',[0.71 0.71 0])

Run simulation on the MNI152 averaged head with default recipe. Pad electrodes will be placed at Fp1 and P4, with default size of [50mm 30mm 3mm] and the long axis will be oriented in the direction specified by the vector [0.71 0.71 0].

Example 12

roast('example/subject1.nii',{'Fp1',1,'FC4',1,'POz',-2},'electype','pad','elecori',{'ap','lr','si'})

Run simulation on subject1 with specified recipe. Pad electrodes will be placed at Fp1, FC4 and POz, with default size of [50mm 30mm 3mm]. The long axis will be oriented in the direction of front to back for the 1st pad, left to right for the 2nd pad, and up to down for the 3rd pad.

Example 13

roast('example/subject1.nii',{'Fp1',1,'FC4',1,'POz',-2},'electype','pad','elecori',[0.71 0.71 0;-0.71 0.71 0;0 0.71 0.71])

Run simulation on subject1 with specified recipe. Pad electrodes will be placed at Fp1, FC4 and POz, with default size of [50mm 30mm 3mm]. The long axis will be oriented in the direction of [0.71 0.71 0] for the 1st pad, [-0.71 0.71 0] for the 2nd pad, and [0 0.71 0.71] for the 3rd pad.

Example 14

roast([],{'Fp1',1,'FC4',1,'POz',-2},'electype',{'pad','disc','pad'},'elecori',[0.71 0.71 0;0 0.71 0.71])

Run simulation on the MNI152 averaged head with specified recipe. A disc electrode will be placed at FC4. Two pad electrodes will be placed at Fp1 and POz, with long axis oriented in the direction of [0.71 0.71 0] and [0 0.71 0.71], respectively.

Example 15

roast([],{'Fp1',1,'FC4',1,'POz',-2},'electype',{'pad','disc','pad'},'elecori',{'ap',[],[0 0.71 0.71]})

Run simulation on the MNI152 averaged head with specified recipe. A disc electrode will be placed at FC4. Two pad electrodes will be placed at Fp1 and POz, with long axis oriented in the direction of front-back and [0 0.71 0.71], respectively.

Example 16

roast('example/subject1.nii',[],'T2','example/subject1_T2.nii')

Run simulation on subject1 with default recipe. The T2 image will be used for segmentation as well.

Example 17

roast('example/subject1.nii',[],'multipriors','on')

Run simulation on subject1 with default recipe. Segmentaion will be done by a deep convolutional neural network known as the MultiPriors.

Example 18

roast([],[],'meshoptions',struct('radbound',4,'maxvol',8))

Run simulation on the MNI152 averaged head with default recipe. Two of the mesh options are customized.

Example 19

roast([],[],'simulationTag','roastDemo')

Give the default run of ROAST a tag as 'roastDemo'.

Example 20

roast('example/subject1.nii',[],'resampling','on')

Run simulaiton on subject1 with default recipe, but resample the MRI of subject1 to 1mm isotropic resolution first (the original MRI of subject1 has resolution of 1mm by 0.99mm by 0.99mm).

Example 21

roast([],{'Exx19',1,'C4',-1},'zeropadding',60,'simulationTag','paddingExample')

Run simulation on the MNI152 averaged head, but add 60 empty slices on each of the six directions to the MRI first, to allow placement of electrode Exx19, which is outside of the MRI (i.e., several centimeters below the most bottom slice of the MRI). This zeropadding also will generate the segmentation of the lower part of the head, thanks to the extended TPM coming along with ROAST. You can visually check this by

reviewRes([],'paddingExample','all')

If you run this without zero-padding first, you'll get strange results. Note it is always a good practice to add empty slices to the MRI if you want to place electrodes close to, or even out of, the image boundary. ROAST can detect if part or all of your electrode goes out of image boundary, but sometimes it cannot tell (it's not that smart yet :-). So do a 'zeroPadding' of 10 to start with, and if you're not happy with the results, just increase the amount of zero-padding. But the best solution is to get an MRI that covers the area where you want to place the electrodes.

Example 22

roast([],{'Fp1',1,'FC4',1,'POz',-2},'conductivities',struct('csf',0.6,'electrode',0.1))

Run simulation on the MNI152 averaged head with specified recipe. The conductivity values of CSF and electrodes are customized. Conductivities of other tissues will use the literature values.

Example 23

roast([],{'Fp1',1,'FC4',1,'POz',-2},'electype',{'pad','disc','pad'},'conductivities',struct('gel',[1 0.3 1],'electrode',[0.1 5.9e7 0.1]))

Run simulation on the MNI152 averaged head with specified recipe. Different conductivities are assigned to pad and disc electrodes. For pad electrodes, 'gel' is given 1 S/m and 'electrode' is 0.1 S/m; for the disc electrode, 'gel' is given 0.3 S/m and 'electrode' is 59000000 S/m. When you control the conductivity values for each electrode individually, keep in mind that the values you put in the vector in 'gel' and 'electrode' field in 'conductivities' option should follow the order of electrodes you put in the 'recipe' argument.

Example 24

All the options can be combined to meet your specific simulation needs.

roast('path/to/your/subject.nii',{'Fp1',0.3,'F8',0.2,'POz',-0.4,'Nk1',0.5,'custom1',-0.6},...
    'electype',{'disc','ring','pad','ring','pad'},...
    'elecsize',{[],[7 9 3],[40 20 4],[],[]},...
    'elecori','ap','T2','path/to/your/t2.nii',...
    'meshoptions',struct('radbound',4,'maxvol',8),...
    'conductivities',struct('csf',0.6,'skin',1.0),...
    'resampling','on','zeropadding',30,...
    'simulationTag','awesomeSimulation')

Now you should know what this will do.

3. How to use roast_target

From ROAST v3.0, users can perform targeted TES (AKA optimized TES) by calling the roast_target() function. To be able to do targeting, you have to first run roast() with leadField as the value for argument recipe, i.e.,

Example 25

roast([],'leadField','simulationTag','MNI152leadField')

This will automatically generate all the lead field data on the MNI152 head required by roast_target() to perform targeting on this head. The candidate electrodes are the 74 electrodes following the 1010 system (2 electrodes on the ear are removed), and you can find their names in a separate text file under ROAST root directory (elec72.loc). Note generating the lead field data will usually take a lot of time (half day to a day depending on the MRI resolution and machine specs). You'll get a warning message the first time you run this asking you for confirmation, so be patient to get the lead field for targeting.

Note you can still config most of the options in ROAST even though you're generating the lead field. Options that are still usable when generating the lead field are: T2, multipriors, meshOptions, simulationTag, resampling, zeroPadding, and conductivities. All the options on electrodes (capType, elecType, elecSize, elecOri) cannot be used and will be set to the defaults for generating the lead field, i.e., capType will be set to 1010, elecType will be set to disc, elecSize will be set to [6mm 2mm], elecOri will be set to [].

You can also generate the lead field for the New York head, which will take even more time as the solving of the New York head (0.5 mm resolution) takes twice the time of solving a 1-mm head model.

Example 26

roast('nyhead','leadField','simulationTag','nyheadLeadField')

If the input MRI has 1 mm isotropic resolution, then results from using leadField as the recipe can be loaded into the Soterix software HD-Explore and HD-Targets. If the input MRI does not have 1 mm isotropic resolution but you turn on the resampling option, the generated lead field can also be readable by Soterix software.

Example 27

roast('example/subject1.nii','leadField','resampling','on','simulationTag','subj1LeadFieldForSoterix')

This will generate the lead field for subject1. Also since the MRI resolution is resampled into 1 mm isotropic, the results can be loaded into Soterix software.

3.1 Synopsis of roast_target

roast_target(subj,simTag,targetCoord,varargin)

subj: file name of the MRI of the subject that you want to run targeting. This follows the same syntax as the roast() function. Make sure that you use the original MRI that you used to run roast() for generating the lead field, even if the original MRI is not in RAS orientation, or you turned on the 'resampling' or did 'zeroPadding' option when running roast() (see Example 30), as roast_target() will take care of all the transforms of MRI data (re-orientation into RAS, resampling or zero-padding).

simTag: the simulationTag that you used in roast() when generating the lead field. For example, if you generated the lead field for the MNI152 head following Example 25, you will use the tag you entered in that example ('MNI152leadField') for this simTag in roast_target(). See Example 28. Similarly, if you run Example 26 first, you can run Example 29 for targeting on the New York head.

targetCoord: the coordinates of the locations in the brain that you want to target. You can do either single location or multiple locations, by putting the coordinates into a N-by-3 matrix, where N is the number of target locations. The coordinates can be either the voxel coordinates or the MNI coordinates, specified by the option coordType (see below). If you don't provide any target location, it defaults to the MNI coordinates of the left primary motor cortex ([-48 -8 50]).

varargin: Options for roast_target(). This follows the same syntax as the roast() function.

'coordType' -- the coordinate space where the target coordinates reside.
'MNI' (default) | 'voxel'
You can tell roast_target() the target locations in either the MNI coordinates or the voxel coordinates. If you use the voxel coordinates, you can use a free program called MRIcro to load the MRI (note do NOT use MRIcron for this as MRIcron will not give you the true voxel coordinates) and click the locations in the brain where you want to target. MRIcro will return the voxel coordinates of the locations you click. Make sure that you use the original MRI that you used to run roast() to click for the voxel coordinates, even if the original MRI is not in RAS orientation, or you turned on the 'resampling' or did 'zeroPadding' option when running roast() to generate the lead field (see Example 30), as roast_target() will take care of all the transforms of MRI data (re-orientation into RAS, resampling or zero-padding).

'optType' -- the specific algorithm used to perform the targeted TES.
'unconstrained-wls' | 'wls-l1' | 'wls-l1per' | 'unconstrained-lcmv' | 'lcmv-l1' | 'lcmv-l1per' | 'max-l1' (default) | 'max-l1per'
You can do either max-focality or max-intensity optimization for TES. Each of the algorithms are explained below. For further details, please refer to this paper. If you want to do multi-focal targeting, it is recommended to use the 'wls-l1' algorithm, see Example 31 to Example 34 and this paper for details.

  • Max-focality algorithms
    • 'unconstrained-wls': unconstrained weighted least squares. This is for max-focality without any constraint on injected current intensities.
    • 'wls-l1': weighted least squares with L1-norm constraint on injected current intensities. The L1-norm constraint enforces the total injected current not beyond 4 mA (2 mA injected into the head and 2 mA coming out of the head).
    • 'wls-l1per': weighted least squares with L1-norm constraint on injected current intensities, with additional L1-norm constraint on each individual electrode. Aside from restricting the total injected current to be below 4 mA, this also restricts the current at each electrode not exceeding 1 mA.
    • 'unconstrained-lcmv': unconstrained LCMV (linearly constrained minimum variance). This is for max-focality without any constraint on injected current intensities.
    • 'lcmv-l1': LCMV with L1-norm constraint on injected current intensities to be smaller than 4 mA.
    • 'lcmv-l1per': LCMV with L1-norm constraint on injected current intensities below 4 mA, with additional L1-norm constraint on each individual electrode to be below 1 mA.
  • Max-intensity algorithms
    • 'max-l1': maximum intensity with L1-norm constraint on injected current intensities to be smaller than 4 mA.
    • 'max-l1per': maximum intensity with L1-norm constraint on injected current intensities below 4 mA., with additional L1-norm constraint on each individual electrode. This can be used together with the option 'elecNum' (see below), to specify the number of electrodes used. This is because 'max-l1' always gives a solution consists of 2 electrodes, with each one having 2 mA flowing through. If you also constrain the current through each electrode to be 1 mA maximum, then the program will split the 1 electrode with 2 mA current into 2 electrodes with each one having 1 mA flowing through, leading to a 4-electrode solution. You can specify how many electrodes you want by using 'elecNum' when you choose 'max-l1per', see Example 35.

'orient' -- the desired orientation of the electric field (i.e., the direction of the current flow) at the target locations.
'radial-in' (default) | 'radial-out' | 'right' | 'left' | 'anterior' | 'posterior' | 'right-anterior' | 'right-posterior' | 'left-anterior' | 'left-posterior' | 'optimal' | orientation vector of your choice
The 'radial-in' means the desired direction of the optimized electric field will point radial inwards to the brain center (whose MNI coordinates is [0 0 0]). Other orientation keywords are self-explanatory. The 'optimal' direction is the direction determined by the program that maximizes the electric field magnitude, see Example 29 and this paper for details. You can also provide the orientation by a customized vector, e.g. [1 1 1], see Example 30. If you provide more than 1 target location, you can specify different orientations at each target, by providing a cell string of different keywords (Example 31) or putting customized orientation vectors into an N-by-3 matrix, where N is the number of target locations (Example 32). You can also mix pre-defined orientation keywords with customized orientation vectors, see Example 33. But you cannot mix the 'optimal' orientation with other orientations.

'desiredIntensity' -- the desired electric field intensity at target location (in V/m), only applies to the max-focality algorithms (the keywords with 'wls' or 'lcmv'), defaults to 1 V/m. If you provide more than 1 target location, you cannot specify different desired intensities at each target.

'elecNum' -- the desired number of electrodes in the optimal montage when using the algorithm 'max-l1per'.
This option only applies when 'optType' is set to 'max-l1per'. Please provide an even number of at least 4 to this option. The default is 4. See Example 35.

'targetRadius' -- advanced option of roast_target(), for controlling the size of each target area. Assuming the target area is a sphere, this gives the radius (in mm) of that sphere. Defaults to 2 mm. If you get the error saying "No nodes found near target", then you should increase the value of this option.

'k' -- advanced option of roast_target(), for adjusting the weights in the weighted least squares algorithm, so this option only applies to 'unconstrained-wls', 'wls-l1' and 'wls-l1per'. The default value is 0.2. If you want more focality but do not care about the intensity of the electric field at the target locations, set 'k' to be low; on the other hand, a high 'k' value will try to attain the desired intensity at the target locations but will not give you that focal electric field. Please refer to this paper for details. Also you may want to decrease 'k' if you want to do multi-focal targeting. See Example 31 to Example 34.

'targetingTag' -- a unique tag that identifies each run of targeting.
dateTime string (default) | user-provided string
This tag is used by roast_target() for managing the data generated from each run of targeting. It's like the 'simulationTag' in roast(), which can identify if a certain run of targeting has been already done. If yes, it will just load the results to save time. You can leave this option empty so that roast_target() will just use the date and time as the unique tag for the targeting. Or you can provide your preferred tag for a specific targeting (Example 33 and Example 34), then you can find it more easily later. Also all the targeting history with options info and results are saved in the log file (named as "subjName_targetLog"), parsed by the targeting tags.

3.2 Examples on roast_target

Example 28

roast_target([],'MNI152leadField')

If you have run Example 25, now you can perform targeting on the MNI152 head using the simulation tag you used in Example 24. Here in this example, we try to guide the current flow to hit the default target location (left primary motor cortex ([-48 -8 50])) with maximal intensity ('max-l1' algorithm used) along the 'radial-in' direction. All the options are in their defaults.

Example 29

roast_target('nyhead','nyheadLeadField',[-48 -8 50;48 -8 50],'optType','lcmv-l1','orient','optimal')

If you have run Example 26, now you can perform targeting on the New York head using the simulation tag you used in Example 25. Here in this example, we aim to target both the left and right primary motor cortex with maximal focality ('lcmv-l1' algorithm used), and the program will also search for the 'optimal' direction where the magnitude of the electric field at target locations are maximized among all possible directions.

Example 30

roast_target('example/subject1.nii','subj1LeadFieldForSoterix',[154 74 156],'coordType','voxel','orient',[1 1 1])

If you have run Example 27, now you can perform targeting on subject1 using the simulation tag you used in Example 26. Here we want to target a location with voxel coordinates. Even though we have turned on the 'resampling' option in Example 26, we should still use the original MRI ('example/subject1.nii') to click for the voxel coordinates in MRIcro, instead of using the re-sampled version ('example/subject1_1mm.nii'), as ROAST takes care of all the transforms applied on the MRI automatically (re-orientation into RAS, resampling or zero-padding). The algorithm used is the default ('max-l1') to maximize the intensity at the target along a customized orientation ([1 1 1]).

Example 31

roast_target([],'MNI152leadField',[-48 -8 50;48 -8 50],'orient',{'right','left'},'optType','wls-l1','k',0.002)

Run targeting on the MNI152 head at both the left and right primary motor cortex with maximal focality ('wls-l1' algorithm used). Desired orientation at the first target (left primary motor cortex) is 'right', and at the second target (right primary motor cortex) is 'left'. A smaller value of 'k' is used for better focality of electric field at the two targets.

Example 32

roast_target([],'MNI152leadField',[-48 -8 50;48 -8 50],'orient',[1 1 1;-1 -1 -1],'optType','wls-l1','k',0.002)

Same as Example 31, but with customized orientations at the two target locations.

Example 33

roast_target([],'MNI152leadField',[52 184 72;25 80 72;139 171 72],'coordType','voxel','optType','wls-l1','k',0.002,'orient',{'radial-in',[-1 1 1],'posterior'},'targetingTag','mixed_orient')

Run targeting on the MNI152 head at three locations indicated by their voxel coordinates (note obtained from original MRI, 'example/MNI152_T1_1mm.nii'), with maximal focality ('wls-l1' algorithm used). Desired orientation at the targets are 'radial-in', customized vector ([-1 1 1]), and 'posterior', respectively. A smaller value of 'k' is used for better focality of electric field at the targets. This targeting run is also tagged as 'mixed_orient'.

In Example 33, you can also leave the 'orient' option blank, so then the desired orientations at the three targets will all be 'radial-in', i.e.

Example 34

roast_target([],'MNI152leadField',[52 184 72;25 80 72;139 171 72],'coordType','voxel','optType','wls-l1','k',0.002,'targetingTag','3targets_radialIn')

Example 35

roast_target([],'MNI152leadField',[],'optType','max-l1per','elecNum',8)

Run targeting on the MNI152 head at the default target location (left primary motor cortex) with maximal intensity ('max-l1per' algorithm used) along the 'radial-in' direction. Besides the L1-norm constraint on the total injected current (4 mA), additional L1-norm constraint on each individual electrode is also enforced. 8 electrodes are specified in the optimal montage, so that the injected current at each individual electrode will not exceed 4 mA / 8 = 0.5 mA. Note by adding the L1-norm constraint on each individual electrode, you may lose electric field intensity at the target.

4. More notes on the capInfo.xlsx file

A lot of info are hidden in the fancy capInfo.xlsx file under the ROAST root directory. There you can find the comprehensive layouts of the 10-05, BioSemi, and EGI HCGSN systems (with my personal drawings), and also visually-striking 3D renderings of the New York head with these electrodes placed on. So make sure to check it out.

5. Outputs of ROAST software

5.1 Outputs of roast

roast() records all the simulation history in a text file named as "subjName_roastLog", where you can find for each simulation (identified by its unique 'simulationTag') the detailed values of all the options.

The simulation data are mainly output in the following formats.

Figure outputs

ROAST outputs 7 or 8 figures for quick visualization of the simulation results. These figures include the slice view of the MRI (T1 and/or T2) and the segmentation; 3D rendering of the computed voltage and electric field distribution; and the slice view of the voltage and electric field. Note the slice view is always in the model voxel space, and the 3D rendering displays the data in the world space. In the slice view you can see both the voxel and MNI coordinates of any point you click in the slices, indicated by a grayish/black circle.

Outputs in Matlab format

ROAST saves the results as "subjName_simulationTag_roastResult.mat", where 3 variables are available:

vol_all: the voltage at each voxel in the model voxel space, unit in mV.

ef_all: the electric field vector at each voxel in the model voxel space, unit in V/m. This variable includes 3 volumes, representing the x-, y-, and z-component of the electric field.

ef_mag: the magnitude of the electric field at each voxel in the model voxel space, unit in V/m.

Outputs in NIfTI format

Voltage: "subjName_simulationTag_v.nii", unit in mV.

E-field: "subjName_simulationTag_e.nii", unit in V/m.

E-field magnitude: "subjName_simulationTag_emag.nii", unit in V/m.

Starting from Version 3.5, ROAST also ouputs the MRI and segmentation mask with their headers aligned to the MNI space, i.e., their voxel-to-world mapping directly maps to the MNI space. These two files end with "_MNI.nii" in their file names.

Outputs in text files

Voltage: "subjName_simulationTag_v.pos", unit in mV.

E-field: "subjName_simulationTag_e.pos", unit in V/m.

Note in these text files, voltage and electric field are defined at each mesh node, whose location can be found in the mesh file "subjName_simulationTag.msh" or "subjName_simulationTag.mat". Also note that in these two mesh files the node coordinates are in the model voxel space but with the scaling factors in the MRI header applied, i.e., the unit of the mesh coordinates is millimeter (mm).

5.2 Outputs of roast_target

roast_target() records all the targeting history in a text file named as "subjName_targetLog", where you can find for each targeting (identified by its unique 'targetingTag') the detailed values of all the options. You will notice the target coordinates ('targetCoord') are recorded in the log file in three formats: MNI, original MRI voxel space, and model voxel space. If you provided the MNI coordinates for the targets, then the original MRI voxel coordinates will be shown as "not provided"; if you provided the voxel coordinates for the targets using the original MRI, then the MNI coordinates will be shown as "not provided". The model voxel coordinates are those coordinates that enter the targeting algorithm, after possible transforms applied on the original MRI (re-orienting into RAS, resampling, or zero-padding). Note also that the targeting results are summarized in this log file as well: the optimal montage used, the achieved electric field magnitude, intensity and focality at each target are all recorded.

The results are also output in the following formats.

Figure outputs

roast_target() outputs at least 3 figures for quick visualization of the targeting results. These figures include the optimal montage displayed as a topoplot; 3D rendering of the optimized electric field in the brain induced by the optimal montage; and the slice view of the optimized electric field, with slice cut orthogonally at each of the target locations, indicated by a grayish/black circle. Note the slice view is always in the model voxel space, and the 3D rendering displays the data in the world space. In the slice view you can see both the voxel and MNI coordinates of any point you click in the slices, indicated by a grayish/black circle.

Outputs in Matlab format

roast_target() saves the results as "subjName_targetingTag_targetResult.mat", where a structure variable named 'r' with the following fields are available:

targetCoord: the target coordinates in the model voxel space.
mon: the optimal montage in a format of a vector with each element indicating the injected current intensity (in mA).
montageTxt: the optimal montage in text format.
xopt: the optimized electric field vector induced by the optimal montage at each mesh node, whose location can be found in the mesh file "subjName_simulationTag.msh" or "subjName_simulationTag.mat". Also note that in these two mesh files the node coordinates are in the model voxel space but with the scaling factors in the MRI header applied, i.e., the unit of the mesh coordinates is millimeter (mm).
ef_all: the optimized electric field induced by the optimal montage at each voxel in the model voxel space, unit in V/m. This variable includes 3 volumes, representing the x-, y-, and z-component of the electric field.
ef_mag: the magnitude of the optimized electric field at each voxel in the model voxel space, unit in V/m.
targetMag: the magnitude of the optimized electric field at each target location (in V/m).
targetInt: the intensity of the optimized electric field along the desired orientation at each target location (in V/m).
targetMagFoc: the focality of the magnitude of the optimized electric field at each target location (in cm).

6. Review of simulation data

You can use a main function reviewRes() to review/visualize the simulations/targetings that you already run before using roast() and roast_target(). reviewRes() has a simpler interface so that you do not have to enter all the option values again as you would have to in roast() or roast_target(). Instead, you just need to enter the path to the input MRI and the unique simulation/targeting tag for that MRI.

reviewRes(subj,simTag,tissue,fastRender,tarTag)

subj: path to the MRI that was used for simulations/targeting, defaults to the MRI of the MNI152 averaged head stored under 'example/'.

simTag: the unique tag of each simulation, which can be looked up in the log file of the subject ("subjName_roastLog").

tissue: which tissue to show in the visualization. You can choose from:
white | gray | csf | bone | skin | air | brain (default) | all
'all' means the entire head volume.

fastRender: do fast 3D rendering or not. By default it's fast rendering. If you turn this option off, it'll generate a smoother surface rendering but also needs more time if the mesh is big.

tarTag: the unique tag of each run of targeting, which can be looked up in the log file of the subject ("subjName_targetLog").

6.1 Review of data from roast

Example 36

reviewRes('path/to/your/subject.nii','awesomeSimulation','white')

Review the results generated from running Example 24, where 'awesomeSimulation' is the corresponding 'simulationTag'. This will show the results in the white matter specifically.

Example 37

reviewRes('nyhead','20180611T185950')

Review the results from simulation tagged '20180611T185950' on the New York head, showing the results in the brian by default.

Example 38

reviewRes('example/subject1.nii','20180613T142621','bone',0)

Review the results from simulation tagged '20180613T142621' on subject1, showing the results in the bone specifically, with smoothed surface rendering. If you change 'bone' to 'all', then it'll show the slice views of the results in all the tissues.

Note this function cannot visualize the lead field. If you ran roast() with 'leadField' as your recipe, please go to roast_target() to run targeting. After running targeting, you can visualize the optimized electric field using this function, by providing both the 'simTag' and 'tarTag', see the next section.

6.2 Review of data from roast_target

Example 39

reviewRes([],'MNI152leadField','','','mixed_orient')

Review the results from running targeting 'mixed_orient' in Example 33, showing the results in the brain by default and with fast rendering by default (no smoothing). You can specify 'all' in the third argument so that it'll show the results in the entire head. You can also specify 0 in the fourth argument so that the 3D rendering will be more smooth (but takes more time to render).

7. How to ask questions

Please read the Getting started and the Synopsis section of each main function. It'll only cost you 10 minutes. If you're lazy or tired, just go to Example 24 and Example 33 for quick references. Do not forget to check out the fancy capInfo.xlsx file.

If ROAST crashes, first check if there are any warning messages output in the command window. If there are, check if there are any suggestions in the warning messages and if yes, follow those suggestions. This will usually fix the problems and let you run through to get your model.

If you are still stuck, please subscribe to the ROAST user mailing list by clicking here. You can ask any questions, answer other users' questions and share your experiences there.

8. Acknowledgements

If you use ROAST in your research, please cite these:

Huang, Y., Datta, A., Bikson, M., Parra, L.C., Realistic vOlumetric-Approach to Simulate Transcranial Electric Stimulation -- ROAST -- a fully automated open-source pipeline, Journal of Neural Engineering, Vol. 16, No. 5, 2019 (prefered reference)

Huang, Y., Datta, A., Bikson, M., Parra, L.C., ROAST: an open-source, fully-automated, Realistic vOlumetric-Approach-based Simulator for TES, Proceedings of the 40th Annual International Conference of the IEEE Engineering in Medicine and Biology Society, Honolulu, HI, July 2018

If you use New York head to run simulation, please also cite the following: Huang, Y., Parra, L.C., Haufe, S.,2016. The New York Head - A precise standardized volume conductor model for EEG source localization and tES targeting. NeuroImage,140, 150-162

If you also use the targeting feature (roast_target), please cite these:

Dmochowski, J.P., Datta, A., Bikson, M., Su, Y., Parra, L.C., Optimized multi-electrode stimulation increases focality and intensity at target, Journal of Neural Engineering 8 (4), 046011, 2011

Dmochowski, J.P., Datta, A., Huang, Y., Richardson, J.D., Bikson, M., Fridriksson, J., Parra, L.C., Targeted transcranial direct current stimulation for rehabilitation after stroke, NeuroImage, 75, 12-19, 2013

Huang, Y., Thomas, C., Datta, A., Parra, L.C., Optimized tDCS for Targeting Multiple Brain Regions: An Integrated Implementation. Proceedings of the 40th Annual International Conference of the IEEE Engineering in Medicine and Biology Society, Honolulu, HI, July 2018, 3545-3548

If you also use the MultiPriors for segmentation by turning on the multipriors option, please cite this:

Hirsch, L., Huang, Y., Parra, L.C., Segmentation of MRI head anatomy using deep volumetric networks and multiple spatial priors. Journal of Medical Imaging, Vol. 8, Issue 3, 034001 (June 2021).

ROAST was supported by the NIH through grants R01MH111896, R01MH111439, R01NS095123, R44NS092144, R41NS076123, and by Soterix Medical Inc.

9. Notes

ROAST is NOT backwards compatible in the generated simulation data.

Starting from Version 3.0, ROAST enforces the RAS rule for the input MRI. So if the input MRI is not in RAS orientation, ROAST will re-orient it into RAS (you'll notice this if you just run ROAST on the default subject: the MNI152 averaged head, see Example 1, as it's in LAS orientation). This is part of our efforts to make ROAST compatible with Soterix software HD-Explore and HD-Targets. In the upcoming release of these two software it is possible to load the results output from ROAST.

Starting from Version 3.5, ROAST is able to build models for heads with lesions, thanks to the multipriors option developed by Lukas Hirsch, and integrated into ROAST by Andrew Birnbaum. This model requires a Python Enviornment which should automatically be installed. If an error occurs durring installation, usually due to incompatibilites in enviornment versions, making your own enviornment is relatively simple.

Creating Multipriors Enviornment

Create your own conda enviornment with the proper OS tag.
-Windows: multipriorsEnv 
-Linux:  multipriorsEnvLinux 
-Mac:  multipriorsEnvMac

Download the proper dependencies by using Conda or Pip Install:
-tensorflow
-scikit-image
-nibabel

Place the enviornment into your lib/multipriors folder.

ROAST will not be able to run on Mac computers that have a silicon chip (M1/M2/M3), only intel will work. An update will be added in the future.

If you do not have Matlab, there is a Docker version.

10. License

General Public License version 3 or later. See LICENSE.md for details.

This software uses free packages from the Internet, except Matlab, which is a proprietary software by the MathWorks. You need a valid Matlab license to run this software.

ROAST is considered as an "aggregate" rather than "derived work", based on the definitions in GPL FAQ. The ROAST license only applies to the scripts, documentations, and the individual MRI data under example/ folder in the root directory in this package and excludes those programs stored in the lib/ directory. The software under lib/ follow their respective licenses. This software is only intended for non-commercial use.

(c) Yu (Andy) Huang, Parra Lab at CCNY

[email protected]

April 2024