2023-08-02 21:53:35 +00:00
|
|
|
"""
|
|
|
|
# https://github.com/enhuiz/pytorch-training-utilities
|
|
|
|
"""
|
|
|
|
|
|
|
|
import os
|
|
|
|
import socket
|
|
|
|
|
|
|
|
from functools import cache, wraps
|
|
|
|
from typing import Callable
|
|
|
|
|
|
|
|
def get_free_port():
|
|
|
|
sock = socket.socket()
|
|
|
|
sock.bind(("", 0))
|
|
|
|
return sock.getsockname()[1]
|
|
|
|
|
|
|
|
|
2023-08-05 03:22:15 +00:00
|
|
|
_distributed_initialized = False
|
2023-09-02 17:23:40 +00:00
|
|
|
def init_distributed( fn, *args, **kwargs ):
|
2024-03-01 16:32:35 +00:00
|
|
|
#print("Initializing distributed...")
|
2023-09-02 17:23:40 +00:00
|
|
|
fn(*args, **kwargs)
|
2023-08-05 03:22:15 +00:00
|
|
|
_distributed_initialized = True
|
|
|
|
|
|
|
|
def distributed_initialized():
|
|
|
|
return _distributed_initialized
|
|
|
|
|
2023-08-02 21:53:35 +00:00
|
|
|
@cache
|
|
|
|
def fix_unset_envs():
|
|
|
|
envs = dict(
|
|
|
|
RANK="0",
|
|
|
|
WORLD_SIZE="1",
|
|
|
|
MASTER_ADDR="localhost",
|
|
|
|
MASTER_PORT=str(get_free_port()),
|
|
|
|
LOCAL_RANK="0",
|
|
|
|
)
|
|
|
|
|
|
|
|
for key in envs:
|
|
|
|
value = os.getenv(key)
|
|
|
|
if value is not None:
|
|
|
|
return
|
|
|
|
|
|
|
|
for key, value in envs.items():
|
|
|
|
os.environ[key] = value
|
|
|
|
|
|
|
|
|
|
|
|
def local_rank():
|
|
|
|
return int(os.getenv("LOCAL_RANK", 0))
|
|
|
|
|
|
|
|
|
|
|
|
def global_rank():
|
|
|
|
return int(os.getenv("RANK", 0))
|
|
|
|
|
2023-08-14 03:56:28 +00:00
|
|
|
def world_size():
|
|
|
|
return int(os.getenv("WORLD_SIZE", 1))
|
|
|
|
|
2023-08-02 21:53:35 +00:00
|
|
|
|
|
|
|
def is_local_leader():
|
|
|
|
return local_rank() == 0
|
|
|
|
|
|
|
|
|
|
|
|
def is_global_leader():
|
|
|
|
return global_rank() == 0
|
|
|
|
|
|
|
|
|
|
|
|
def local_leader_only(fn=None, *, default=None) -> Callable:
|
|
|
|
def wrapper(fn):
|
|
|
|
@wraps(fn)
|
|
|
|
def wrapped(*args, **kwargs):
|
|
|
|
if is_local_leader():
|
|
|
|
return fn(*args, **kwargs)
|
|
|
|
return default
|
|
|
|
|
|
|
|
return wrapped
|
|
|
|
|
|
|
|
if fn is None:
|
|
|
|
return wrapper
|
|
|
|
|
|
|
|
return wrapper(fn)
|
|
|
|
|
|
|
|
|
|
|
|
def global_leader_only(fn: Callable | None = None, *, default=None) -> Callable:
|
|
|
|
def wrapper(fn):
|
|
|
|
@wraps(fn)
|
|
|
|
def wrapped(*args, **kwargs):
|
|
|
|
if is_global_leader():
|
|
|
|
return fn(*args, **kwargs)
|
|
|
|
return default
|
|
|
|
|
|
|
|
return wrapped
|
|
|
|
|
|
|
|
if fn is None:
|
|
|
|
return wrapper
|
|
|
|
|
|
|
|
return wrapper(fn)
|