I believe this kind of implementation will result in loosing values of alpha
and k
as well as slow_weights
.
True, but since alpha & k are constant for a given training, I havenât had to deal with that for now (and havenât yet played around with alpha and k).
I considered passing alpha and k to the state_dict of base_optimizer
to solve this. But what was the exact issue you had with Pickle interaction using your implementation?
EDIT: actually, you gave me an idea, I just pushed a commit to reconciliate the state getter & loader as well as having alpha, k! Cheers for that @grankin
The issue is that I get different result if I do save/load and if I donât. I think the proper behaviour for save/load is like if it didnât happened.
The Pickle loader donât call the __init__
constructor and alpha and k would be uninitialised after load
. And the slow_weight are lost.
I guess you did, but better ask: have you set a seed for this check?
I see, would you have the code reference for this pickle loader? so that I cant take a look
I use seed and I can get deterministic results between run.
Iâve used fast.aiâs load
https://github.com/fastai/fastai/blob/master/fastai/basic_train.py#L262
Which in turn uses Pytorch load
torch.load
And Pytorch uses a Pythonâs Pickle https://docs.python.org/3/library/pickle.html
Oh, I thought you were talking about an alternative loading method. As you pointed out the param duplication, I looked more closely the PyTorch optimizer implementation and made a few changes to my implementation.
It should solve the duplication issue as well as improve support for base optimizer method support (so thatâs itâs a proper wrapper )
Let me know if the commit helps!
Most of the Lookahead / Ranger impl have some issues with state dict save/load and also adding parameters after Optimizer creation via add_param_group
causing a crash.
Iâve gone through a few iterations starting from https://github.com/alphadl/lookahead.pytorch (which is closer to correct than the lonePatient impl).
Currently state is here, tested resumes with different optimizers. Could still be issues but I think itâs close
EDIT: Also, added support in this one to resume a checkpoint with Lookahead(OptA) that was created with just OptA
Wow, thatâs terrific job, works like a charm! I did save/load and got exact same validation loss.
slow_weights are now stored in param_groups, I like that trick!
Yes, I figured it would give more coherence to the implementation since we inherit from Optimizer
. And it allows us to use inherited methods to reduce code base The only method that Iâm not overwriting is a private one to ensure smooth param_group addition for slow weights.
Also, you might want to check the discussion on this thread. I added a param synchronization method for external calls so that users can choose the state they want for their model to be evaluated. My current version is available here!
Cheers
Your procedure to use ranger is not working fro me i am getting Typeerror:module object is not callable when i am running this line â>optar=partial(Ranger).
Thanks for your work !
Using Ranger on my model, I did a save/load , and start the training again. And the training loss behaves in a completely different way after this save/load step (training speed decreases).
Anyone has the same problem ? I think the saved optimizer is not saved correctly. I have better performance if I specify with_opt=False
when I load the model.
Thanks for the feedback! It sounds like we are dragging around duplicate slow weightsâŚI will take a look and try to fix!
I didnât get a chance to test it, but I believe the fix is to simply leverage what @rwightman did and move the slow weights into a state param group (which as usual, is brilliant coding by him).
That way they are re-loaded properly and should correct this issue.
Will try and do that tomorrow but at least I believe I know the issue and by copying @rwightman excellent idea, should fix.
I think the optimizer does not work with pretrained models when the model has different layer groups. For some reason, it stops after one epoch. Could you please look into this?
Iâm testing an update now that should handle layer groups. What model are you using and Iâll test a run with the fix.
Thanks!
I am using an EfficientNetB3 from the Luke Melas repository with the following split:
learn.split( lambda m: (m._conv_head,) )
Iâve posted a new version of Ranger - it has improved support for layer groups and is a much tighter codebase all around (one pass handling at param level, no repeat loops, slow weights moved into state dict, etc).
Can you please see if that resolves your issue?
New version 9.3.19
*Also thanks to @rwightman as I leveraged some of his code ideas related to putting the slow weights into a state dictionary vs how lonepatient originally did it.
Iâm working to integrate @fgfm idea regarding partial sync as well next.
Thanks! When I get the chance, I will try it out with layer groups and let you know!
Is there any kind of early consensus around how to handle .fit_one_cycle()
in combination with RAdam
or Ranger
yet?