Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Generalized Dice Loss - Reduction="none" + batch=True #5466

Closed
feliksh opened this issue Nov 3, 2022 · 2 comments · Fixed by #5468
Closed

Generalized Dice Loss - Reduction="none" + batch=True #5466

feliksh opened this issue Nov 3, 2022 · 2 comments · Fixed by #5468

Comments

@feliksh
Copy link

feliksh commented Nov 3, 2022

Describe the bug
GeneralizedDiceLoss(..., reduction="none", batch=True) does not return the expected shape.

To Reproduce
Steps to reproduce the behavior:

Given tensors of shape:
  network_prediction -> B x C x H x W x D
  labels -> B x 1 x H x W x D
  with B=Batch_size, C=Channels/Classes, Height, Width, Depth

Correct behaviour, since I am asking for no reduction and batch=True:

loss_fun1 = DiceLoss(to_onehot_y=True, softmax=True, reduction="none", batch=True)
loss1 = loss_fun1(network_prediction, labels)

loss1 -> [C x 1 x 1 x 1]

While if you do the same with GeneralizedDiceLoss:

loss_fun2 = GeneralizedDiceLoss(to_onehot_y=True, softmax=True, reduction="none", batch=True)
loss2 = loss_fun1(network_prediction, labels)

loss2 -> 1 x 1 x 1 x 1

which is not correct, since again you are asking for no reduction and batch=True.

The relevant piece of code which seems to lose the correct shape is in
https://github.com/Project-MONAI/MONAI/blob/dev/monai/losses/dice.py#L360

final_reduce_dim = 0 if self.batch else 1
numer = 2.0 * (intersection * w).sum(final_reduce_dim, keepdim=True) + self.smooth_nr
denom = (denominator * w).sum(final_reduce_dim, keepdim=True) + self.smooth_dr
f: torch.Tensor = 1.0 - (numer / denom)

where numer and denom are collapsed to shape (1,) since both intersection,denominator and w are of shape (C,),
thus .sum(final_reduce_dim, keepdim=True) does collapse the shape (C,) into (1,).

I believe the solution would be to not let those tensors be of singleton shape (C,) but instead keep them at (1, C) or (C, 1)

Environment

Python 3.7.9,
Monai 0.9.1,
PyTorch 1.11.0

KumoLiu added a commit to KumoLiu/MONAI that referenced this issue Nov 4, 2022
Signed-off-by: KumoLiu <yunl@nvidia.com>
@KumoLiu KumoLiu mentioned this issue Nov 4, 2022
7 tasks
@KumoLiu
Copy link
Contributor

KumoLiu commented Nov 4, 2022

Hi @feliksh, thanks for pointing it out, I just create a PR #5468 to fix it.

@wyli wyli closed this as completed in #5468 Nov 4, 2022
wyli pushed a commit that referenced this issue Nov 4, 2022
Signed-off-by: KumoLiu <yunl@nvidia.com>

Fixes #5466.

### Description
In `GeneralizedDiceLoss`, now it do channel reduction before reduction,
remove channel reduction in this PR.

https://github.com/Project-MONAI/MONAI/blob/c38d503a587f1779914bd071a1b2d66a6d9080c2/monai/losses/dice.py#L360-L363

### Types of changes
<!--- Put an `x` in all the boxes that apply, and remove the not
applicable items -->
- [x] Non-breaking change (fix or new feature that would not break
existing functionality).
- [ ] Breaking change (fix or new feature that would cause existing
functionality to change).
- [x] New tests added to cover the changes.
- [ ] Integration tests passed locally by running `./runtests.sh -f -u
--net --coverage`.
- [ ] Quick tests passed locally by running `./runtests.sh --quick
--unittests --disttests`.
- [ ] In-line docstrings updated.
- [ ] Documentation updated, tested `make html` command in the `docs/`
folder.

Signed-off-by: KumoLiu <yunl@nvidia.com>
wyli pushed a commit that referenced this issue Nov 6, 2022
Signed-off-by: KumoLiu <yunl@nvidia.com>

Fixes #5466.

### Description
In `GeneralizedDiceLoss`, now it do channel reduction before reduction,
remove channel reduction in this PR.

https://github.com/Project-MONAI/MONAI/blob/c38d503a587f1779914bd071a1b2d66a6d9080c2/monai/losses/dice.py#L360-L363

### Types of changes
<!--- Put an `x` in all the boxes that apply, and remove the not
applicable items -->
- [x] Non-breaking change (fix or new feature that would not break
existing functionality).
- [ ] Breaking change (fix or new feature that would cause existing
functionality to change).
- [x] New tests added to cover the changes.
- [ ] Integration tests passed locally by running `./runtests.sh -f -u
--net --coverage`.
- [ ] Quick tests passed locally by running `./runtests.sh --quick
--unittests --disttests`.
- [ ] In-line docstrings updated.
- [ ] Documentation updated, tested `make html` command in the `docs/`
folder.

Signed-off-by: KumoLiu <yunl@nvidia.com>
@feliksh
Copy link
Author

feliksh commented Nov 7, 2022

That's fantastic, impressed by the quickness. Well done, thank you!

wyli added a commit that referenced this issue Jul 26, 2023
Fixes #6765

### Description
as discussed in #6765, when `batch=True` the loss should still return 1
aggregated value instead of C channels.
#5466 is not actually
achievable with this formulation.

### Types of changes
<!--- Put an `x` in all the boxes that apply, and remove the not
applicable items -->
- [x] Non-breaking change (fix or new feature that would not break
existing functionality).
- [ ] Breaking change (fix or new feature that would cause existing
functionality to change).
- [ ] New tests added to cover the changes.
- [ ] Integration tests passed locally by running `./runtests.sh -f -u
--net --coverage`.
- [ ] Quick tests passed locally by running `./runtests.sh --quick
--unittests --disttests`.
- [ ] In-line docstrings updated.
- [ ] Documentation updated, tested `make html` command in the `docs/`
folder.

---------

Signed-off-by: Wenqi Li <wenqil@nvidia.com>
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging a pull request may close this issue.

2 participants