Skip to content

Commit 5f332e8

Browse files
authored
Separate goodvoxels mask creation from fsLR resampling (#3170)
Closes #3169. This could just be a first step toward moving one or more of these workflows from fMRIPrep to niworkflows. - Move `init_goodvoxels_bold_mask_wf` call out of `init_bold_fsLR_resampling_wf`. This will allow downstream tools (e.g., ASLPrep) to generate the goodvoxels mask from one file (the ASL time series) and apply it to others (e.g., CBF maps). - Clean up `init_bold_fsLR_resampling_wf` (e.g., remove unused imports and add all parameters to docstring). - Move BOLD-to-anatomical resampling workflow call from `resampling` level to `full` level, since it doesn't produce any outputs.
2 parents d24f585 + fb1c989 commit 5f332e8

File tree

2 files changed

+68
-72
lines changed

2 files changed

+68
-72
lines changed

fmriprep/workflows/bold/base.py

+52-29
Original file line numberDiff line numberDiff line change
@@ -31,8 +31,6 @@
3131
"""
3232
import typing as ty
3333

34-
import nibabel as nb
35-
import numpy as np
3634
from nipype.interfaces import utility as niu
3735
from nipype.pipeline import engine as pe
3836
from niworkflows.utils.connections import listify
@@ -295,44 +293,20 @@ def init_bold_wf(
295293
fieldmap_id=fieldmap_id,
296294
omp_nthreads=omp_nthreads,
297295
)
298-
bold_anat_wf = init_bold_volumetric_resample_wf(
299-
metadata=all_metadata[0],
300-
fieldmap_id=fieldmap_id if not multiecho else None,
301-
omp_nthreads=omp_nthreads,
302-
mem_gb=mem_gb,
303-
name='bold_anat_wf',
304-
)
305-
bold_anat_wf.inputs.inputnode.resolution = "native"
306296

307297
workflow.connect([
308298
(inputnode, bold_native_wf, [
309299
("fmap_ref", "inputnode.fmap_ref"),
310300
("fmap_coeff", "inputnode.fmap_coeff"),
311301
("fmap_id", "inputnode.fmap_id"),
312302
]),
313-
(inputnode, bold_anat_wf, [
314-
("t1w_preproc", "inputnode.target_ref_file"),
315-
("t1w_mask", "inputnode.target_mask"),
316-
("fmap_ref", "inputnode.fmap_ref"),
317-
("fmap_coeff", "inputnode.fmap_coeff"),
318-
("fmap_id", "inputnode.fmap_id"),
319-
]),
320303
(bold_fit_wf, bold_native_wf, [
321304
("outputnode.coreg_boldref", "inputnode.boldref"),
322305
("outputnode.bold_mask", "inputnode.bold_mask"),
323306
("outputnode.motion_xfm", "inputnode.motion_xfm"),
324307
("outputnode.boldref2fmap_xfm", "inputnode.boldref2fmap_xfm"),
325308
("outputnode.dummy_scans", "inputnode.dummy_scans"),
326309
]),
327-
(bold_fit_wf, bold_anat_wf, [
328-
("outputnode.coreg_boldref", "inputnode.bold_ref_file"),
329-
("outputnode.boldref2fmap_xfm", "inputnode.boldref2fmap_xfm"),
330-
("outputnode.boldref2anat_xfm", "inputnode.boldref2anat_xfm"),
331-
]),
332-
(bold_native_wf, bold_anat_wf, [
333-
("outputnode.bold_minimal", "inputnode.bold_file"),
334-
("outputnode.motion_xfm", "inputnode.motion_xfm"),
335-
]),
336310
]) # fmt:skip
337311

338312
boldref_out = bool(nonstd_spaces.intersection(('func', 'run', 'bold', 'boldref', 'sbref')))
@@ -400,6 +374,35 @@ def init_bold_wf(
400374
if config.workflow.level == "resampling":
401375
return workflow
402376

377+
# Resample to anatomical space
378+
bold_anat_wf = init_bold_volumetric_resample_wf(
379+
metadata=all_metadata[0],
380+
fieldmap_id=fieldmap_id if not multiecho else None,
381+
omp_nthreads=omp_nthreads,
382+
mem_gb=mem_gb,
383+
name='bold_anat_wf',
384+
)
385+
bold_anat_wf.inputs.inputnode.resolution = "native"
386+
387+
workflow.connect([
388+
(inputnode, bold_anat_wf, [
389+
("t1w_preproc", "inputnode.target_ref_file"),
390+
("t1w_mask", "inputnode.target_mask"),
391+
("fmap_ref", "inputnode.fmap_ref"),
392+
("fmap_coeff", "inputnode.fmap_coeff"),
393+
("fmap_id", "inputnode.fmap_id"),
394+
]),
395+
(bold_fit_wf, bold_anat_wf, [
396+
("outputnode.coreg_boldref", "inputnode.bold_ref_file"),
397+
("outputnode.boldref2fmap_xfm", "inputnode.boldref2fmap_xfm"),
398+
("outputnode.boldref2anat_xfm", "inputnode.boldref2anat_xfm"),
399+
]),
400+
(bold_native_wf, bold_anat_wf, [
401+
("outputnode.bold_minimal", "inputnode.bold_file"),
402+
("outputnode.motion_xfm", "inputnode.motion_xfm"),
403+
]),
404+
]) # fmt:skip
405+
403406
# Full derivatives, including resampled BOLD series
404407
if nonstd_spaces.intersection(('anat', 'T1w')):
405408
ds_bold_t1_wf = init_ds_volumes_wf(
@@ -507,7 +510,11 @@ def init_bold_wf(
507510
]) # fmt:skip
508511

509512
if config.workflow.cifti_output:
510-
from .resampling import init_bold_fsLR_resampling_wf, init_bold_grayords_wf
513+
from .resampling import (
514+
init_bold_fsLR_resampling_wf,
515+
init_bold_grayords_wf,
516+
init_goodvoxels_bold_mask_wf,
517+
)
511518

512519
bold_MNI6_wf = init_bold_volumetric_resample_wf(
513520
metadata=all_metadata[0],
@@ -518,12 +525,29 @@ def init_bold_wf(
518525
)
519526

520527
bold_fsLR_resampling_wf = init_bold_fsLR_resampling_wf(
521-
estimate_goodvoxels=config.workflow.project_goodvoxels,
522528
grayord_density=config.workflow.cifti_output,
523529
omp_nthreads=omp_nthreads,
524530
mem_gb=mem_gb["resampled"],
525531
)
526532

533+
if config.workflow.project_goodvoxels:
534+
goodvoxels_bold_mask_wf = init_goodvoxels_bold_mask_wf(mem_gb["resampled"])
535+
536+
workflow.connect([
537+
(inputnode, goodvoxels_bold_mask_wf, [("anat_ribbon", "inputnode.anat_ribbon")]),
538+
(bold_anat_wf, goodvoxels_bold_mask_wf, [
539+
("outputnode.bold_file", "inputnode.bold_file"),
540+
]),
541+
(goodvoxels_bold_mask_wf, bold_fsLR_resampling_wf, [
542+
("outputnode.goodvoxels_mask", "inputnode.volume_roi"),
543+
]),
544+
]) # fmt:skip
545+
546+
bold_fsLR_resampling_wf.__desc__ += """\
547+
A "goodvoxels" mask was applied during volume-to-surface sampling in fsLR space,
548+
excluding voxels whose time-series have a locally high coefficient of variation.
549+
"""
550+
527551
bold_grayords_wf = init_bold_grayords_wf(
528552
grayord_density=config.workflow.cifti_output,
529553
mem_gb=1,
@@ -573,7 +597,6 @@ def init_bold_wf(
573597
("midthickness_fsLR", "inputnode.midthickness_fsLR"),
574598
("sphere_reg_fsLR", "inputnode.sphere_reg_fsLR"),
575599
("cortex_mask", "inputnode.cortex_mask"),
576-
("anat_ribbon", "inputnode.anat_ribbon"),
577600
]),
578601
(bold_anat_wf, bold_fsLR_resampling_wf, [
579602
("outputnode.bold_file", "inputnode.bold_file"),

fmriprep/workflows/bold/resampling.py

+16-43
Original file line numberDiff line numberDiff line change
@@ -503,7 +503,6 @@ def _calc_lower_thr(in_stats):
503503

504504
def init_bold_fsLR_resampling_wf(
505505
grayord_density: ty.Literal['91k', '170k'],
506-
estimate_goodvoxels: bool,
507506
omp_nthreads: int,
508507
mem_gb: float,
509508
name: str = "bold_fsLR_resampling_wf",
@@ -522,7 +521,6 @@ def init_bold_fsLR_resampling_wf(
522521
523522
from fmriprep.workflows.bold.resampling import init_bold_fsLR_resampling_wf
524523
wf = init_bold_fsLR_resampling_wf(
525-
estimate_goodvoxels=True,
526524
grayord_density='92k',
527525
omp_nthreads=1,
528526
mem_gb=1,
@@ -532,9 +530,6 @@ def init_bold_fsLR_resampling_wf(
532530
----------
533531
grayord_density : :class:`str`
534532
Either ``"91k"`` or ``"170k"``, representing the total *grayordinates*.
535-
estimate_goodvoxels : :class:`bool`
536-
Calculate mask excluding voxels with a locally high coefficient of variation to
537-
exclude from surface resampling
538533
omp_nthreads : :class:`int`
539534
Maximum number of threads an individual process may use
540535
mem_gb : :class:`float`
@@ -546,35 +541,33 @@ def init_bold_fsLR_resampling_wf(
546541
------
547542
bold_file : :class:`str`
548543
Path to BOLD file resampled into T1 space
549-
surfaces : :class:`list` of :class:`str`
550-
Path to left and right hemisphere white, pial and midthickness GIFTI surfaces
551-
morphometrics : :class:`list` of :class:`str`
552-
Path to left and right hemisphere morphometric GIFTI surfaces, which must include thickness
544+
white : :class:`list` of :class:`str`
545+
Path to left and right hemisphere white matter GIFTI surfaces.
546+
pial : :class:`list` of :class:`str`
547+
Path to left and right hemisphere pial GIFTI surfaces.
548+
midthickness : :class:`list` of :class:`str`
549+
Path to left and right hemisphere midthickness GIFTI surfaces.
550+
midthickness_fsLR : :class:`list` of :class:`str`
551+
Path to left and right hemisphere midthickness GIFTI surfaces in fsLR space.
553552
sphere_reg_fsLR : :class:`list` of :class:`str`
554553
Path to left and right hemisphere sphere.reg GIFTI surfaces, mapping from subject to fsLR
555-
anat_ribbon : :class:`str`
556-
Path to mask of cortical ribbon in T1w space, for calculating goodvoxels
554+
cortex_mask : :class:`list` of :class:`str`
555+
Path to left and right hemisphere cortical masks.
556+
volume_roi : :class:`str` or Undefined
557+
Pre-calculated goodvoxels mask. Not required.
557558
558559
Outputs
559560
-------
560561
bold_fsLR : :class:`list` of :class:`str`
561562
Path to BOLD series resampled as functional GIFTI files in fsLR space
562-
goodvoxels_mask : :class:`str`
563-
Path to mask of voxels, excluding those with locally high coefficients of variation
564563
565564
"""
566565
import templateflow.api as tf
567566
from niworkflows.engine.workflows import LiterateWorkflow as Workflow
568567
from niworkflows.interfaces.utility import KeySelect
569568
from smriprep import data as smriprep_data
570-
from smriprep.interfaces.workbench import SurfaceResample
571569

572-
from fmriprep.interfaces.gifti import CreateROI
573-
from fmriprep.interfaces.workbench import (
574-
MetricFillHoles,
575-
MetricRemoveIslands,
576-
VolumeToSurfaceMapping,
577-
)
570+
from fmriprep.interfaces.workbench import VolumeToSurfaceMapping
578571

579572
fslr_density = "32k" if grayord_density == "91k" else "59k"
580573

@@ -589,13 +582,13 @@ def init_bold_fsLR_resampling_wf(
589582
niu.IdentityInterface(
590583
fields=[
591584
'bold_file',
592-
'anat_ribbon',
593585
'white',
594586
'pial',
595587
'midthickness',
596588
'midthickness_fsLR',
597589
'sphere_reg_fsLR',
598590
'cortex_mask',
591+
'volume_roi',
599592
]
600593
),
601594
name='inputnode',
@@ -614,7 +607,7 @@ def init_bold_fsLR_resampling_wf(
614607
)
615608

616609
outputnode = pe.Node(
617-
niu.IdentityInterface(fields=['bold_fsLR', 'goodvoxels_mask']),
610+
niu.IdentityInterface(fields=['bold_fsLR']),
618611
name='outputnode',
619612
)
620613

@@ -689,6 +682,7 @@ def init_bold_fsLR_resampling_wf(
689682
# Resample BOLD to native surface, dilate and mask
690683
(inputnode, volume_to_surface, [
691684
('bold_file', 'volume_file'),
685+
('volume_roi', 'volume_roi'),
692686
]),
693687
(select_surfaces, volume_to_surface, [
694688
('midthickness', 'surface_file'),
@@ -715,27 +709,6 @@ def init_bold_fsLR_resampling_wf(
715709
(joinnode, outputnode, [('bold_fsLR', 'bold_fsLR')]),
716710
]) # fmt:skip
717711

718-
if estimate_goodvoxels:
719-
workflow.__desc__ += """\
720-
A "goodvoxels" mask was applied during volume-to-surface sampling in fsLR space,
721-
excluding voxels whose time-series have a locally high coefficient of variation.
722-
"""
723-
724-
goodvoxels_bold_mask_wf = init_goodvoxels_bold_mask_wf(mem_gb)
725-
726-
workflow.connect([
727-
(inputnode, goodvoxels_bold_mask_wf, [
728-
("bold_file", "inputnode.bold_file"),
729-
("anat_ribbon", "inputnode.anat_ribbon"),
730-
]),
731-
(goodvoxels_bold_mask_wf, volume_to_surface, [
732-
("outputnode.goodvoxels_mask", "volume_roi"),
733-
]),
734-
(goodvoxels_bold_mask_wf, outputnode, [
735-
("outputnode.goodvoxels_mask", "goodvoxels_mask"),
736-
]),
737-
]) # fmt:skip
738-
739712
return workflow
740713

741714

0 commit comments

Comments
 (0)