diff --git a/docs/source-fabric/_static/images/logo.svg b/docs/source-fabric/_static/images/logo.svg index 60efaa29b8bbc..7c0920c73c315 100644 --- a/docs/source-fabric/_static/images/logo.svg +++ b/docs/source-fabric/_static/images/logo.svg @@ -1,12 +1,21 @@ - - - - - + + + + + + + + - - - - + + + + + + + + + + diff --git a/docs/source-fabric/api/accelerators.rst b/docs/source-fabric/api/accelerators.rst new file mode 100644 index 0000000000000..67dffdf5ee8dc --- /dev/null +++ b/docs/source-fabric/api/accelerators.rst @@ -0,0 +1,22 @@ +.. include:: ../links.rst + +############################# +lightning.fabric.accelerators +############################# + + +Accelerators +^^^^^^^^^^^^ + +.. currentmodule:: lightning.fabric.accelerators + +.. autosummary:: + :toctree: ./generated + :nosignatures: + :template: classtemplate.rst + + Accelerator + CPUAccelerator + CUDAAccelerator + MPSAccelerator + TPUAccelerator diff --git a/docs/source-fabric/api/collectives.rst b/docs/source-fabric/api/collectives.rst new file mode 100644 index 0000000000000..21a77342abe25 --- /dev/null +++ b/docs/source-fabric/api/collectives.rst @@ -0,0 +1,23 @@ +.. include:: ../links.rst + +#################################### +lightning.fabric.plugins.collectives +#################################### + +.. warning:: + This is an `experimental `__ feature. + + +Collectives +^^^^^^^^^^^ + +.. currentmodule:: lightning.fabric.plugins.collectives + +.. autosummary:: + :toctree: ./generated + :nosignatures: + :template: classtemplate.rst + + Collective + TorchCollective + SingleDeviceCollective diff --git a/docs/source-fabric/api/environments.rst b/docs/source-fabric/api/environments.rst new file mode 100644 index 0000000000000..f824ed80864b5 --- /dev/null +++ b/docs/source-fabric/api/environments.rst @@ -0,0 +1,25 @@ +.. include:: ../links.rst + +##################################### +lightning.fabric.plugins.environments +##################################### + + +Environments +^^^^^^^^^^^^ + +.. currentmodule:: lightning.fabric.plugins.environments + +.. autosummary:: + :toctree: ./generated + :nosignatures: + :template: classtemplate_noindex.rst + + ~cluster_environment.ClusterEnvironment + ~kubeflow.KubeflowEnvironment + ~lightning.LightningEnvironment + ~lsf.LSFEnvironment + ~mpi.MPIEnvironment + ~slurm.SLURMEnvironment + ~torchelastic.TorchElasticEnvironment + ~xla.XLAEnvironment diff --git a/docs/source-fabric/api/fabric.rst b/docs/source-fabric/api/fabric.rst new file mode 100644 index 0000000000000..41036e3270c7b --- /dev/null +++ b/docs/source-fabric/api/fabric.rst @@ -0,0 +1,18 @@ +.. include:: ../links.rst + +####################### +lightning.fabric.Fabric +####################### + + +Fabric +^^^^^^ + +.. currentmodule:: lightning.fabric.fabric + +.. autosummary:: + :toctree: ./generated + :nosignatures: + :template: classtemplate.rst + + Fabric diff --git a/docs/source-fabric/api/io.rst b/docs/source-fabric/api/io.rst new file mode 100644 index 0000000000000..8253a4b5d0400 --- /dev/null +++ b/docs/source-fabric/api/io.rst @@ -0,0 +1,24 @@ +.. include:: ../links.rst + +########################### +lightning.fabric.plugins.io +########################### + + +.. warning:: + This is an `experimental `__ feature. + + +IO +^^ + +.. currentmodule:: lightning.fabric.plugins.io + +.. autosummary:: + :toctree: ./generated + :nosignatures: + :template: classtemplate.rst + + ~checkpoint_io.CheckpointIO + ~torch_io.TorchCheckpointIO + ~xla.XLACheckpointIO diff --git a/docs/source-fabric/api/loggers.rst b/docs/source-fabric/api/loggers.rst new file mode 100644 index 0000000000000..3c4936c0ac124 --- /dev/null +++ b/docs/source-fabric/api/loggers.rst @@ -0,0 +1,20 @@ +.. include:: ../links.rst + +######################## +lightning.fabric.loggers +######################## + + +Loggers +^^^^^^^ + +.. currentmodule:: lightning.fabric.loggers + +.. autosummary:: + :toctree: ./generated + :nosignatures: + :template: classtemplate.rst + + Logger + CSVLogger + TensorBoardLogger diff --git a/docs/source-fabric/api/precision.rst b/docs/source-fabric/api/precision.rst new file mode 100644 index 0000000000000..94e88e414f19c --- /dev/null +++ b/docs/source-fabric/api/precision.rst @@ -0,0 +1,25 @@ +.. include:: ../links.rst + +################################## +lightning.fabric.plugins.precision +################################## + + +Precision +^^^^^^^^^ + +.. TODO(fabric): include DeepSpeedPrecision + +.. currentmodule:: lightning.fabric.plugins.precision + +.. autosummary:: + :toctree: ./generated + :nosignatures: + :template: classtemplate.rst + + Precision + DoublePrecision + MixedPrecision + TPUPrecision + TPUBf16Precision + FSDPPrecision diff --git a/docs/source-fabric/api/strategies.rst b/docs/source-fabric/api/strategies.rst new file mode 100644 index 0000000000000..b30a2d6ff3c93 --- /dev/null +++ b/docs/source-fabric/api/strategies.rst @@ -0,0 +1,26 @@ +.. include:: ../links.rst + +########################### +lightning.fabric.strategies +########################### + + +Strategies +^^^^^^^^^^ + +.. TODO(fabric): include DeepSpeedStrategy, XLAStrategy + +.. currentmodule:: lightning.fabric.strategies + +.. autosummary:: + :toctree: ./generated + :nosignatures: + :template: classtemplate.rst + + Strategy + DDPStrategy + DataParallelStrategy + FSDPStrategy + ParallelStrategy + SingleDeviceStrategy + SingleTPUStrategy diff --git a/docs/source-fabric/api/utilities.rst b/docs/source-fabric/api/utilities.rst deleted file mode 100644 index 9437317c8ee22..0000000000000 --- a/docs/source-fabric/api/utilities.rst +++ /dev/null @@ -1,88 +0,0 @@ -################ -Fabric Utilities -################ - - -seed_everything -=============== - -This function sets the random seed in important libraries. -In a single line of code, you can seed PyTorch, NumPy, and Python: - -.. code-block:: diff - - + from lightning.fabric import seed_everything - - seed = 42 - - random.seed(seed) - - numpy.random.seed(seed) - - torch.manual_seed(seed) - - torch.cuda.manual_seed(seed) - - + seed_everything(seed) - -The same is also available as a method on the Fabric object if you don't want to import it separately: - -.. code-block:: python - - from lightning.fabric import Fabric - - fabric.Fabric() - fabric.seed_everything(42) - - -In distributed settings, you may need to set a different seed per process, depending on the application. -For example, when generating noise or data augmentations. This is very straightforward: - -.. code-block:: python - - fabric = Fabric(...) - fabric.seed_everything(seed + fabric.global_rank) - - -By default, :meth:`~lightning.fabric.fabric.Fabric.seed_everything` also handles the initialization of the seed in :class:`~torch.utils.data.DataLoader` worker processes: - -.. code-block:: python - - fabric = Fabric(...) - - # By default, we handle DataLoader workers too: - fabric.seed_everything(..., workers=True) - - # Can be turned off: - fabric.seed_everything(..., workers=False) - - ----- - - -print -===== - -Avoid duplicated print statements in the logs in distributed training by using Fabric's :meth:`~lightning.fabric.fabric.Fabric.print` method: - -.. code-block:: python - - print("This message gets printed in every process. That's a bit messy!") - - fabric = Fabric(...) - fabric.print("This message gets printed only in the main process. Much cleaner!") - - ----- - - -is_wrapped -========== - -You can test whether an object (:class:`~torch.nn.Module`, :class:`~torch.optim.Optimizer`, :class:`~torch.utils.data.DataLoader`) was already set up once by Fabric: - -.. code-block:: python - - from lightning.fabric import is_wrapped - - if not is_wrapped(model): - model = fabric.setup(model) - - if not is_wrapped(train_dataloader): - train_dataloader = fabric.setup_dataloaders(train_dataloader) diff --git a/docs/source-fabric/api_reference.rst b/docs/source-fabric/api_reference.rst deleted file mode 100644 index 1e64f7313c87a..0000000000000 --- a/docs/source-fabric/api_reference.rst +++ /dev/null @@ -1,140 +0,0 @@ -.. include:: links.rst - -############# -API Reference -############# - - -Fabric -^^^^^^ - -.. currentmodule:: lightning.fabric.fabric - -.. autosummary:: - :toctree: api/generated - :nosignatures: - :template: classtemplate.rst - - Fabric - - -Accelerators -^^^^^^^^^^^^ - -.. currentmodule:: lightning.fabric.accelerators - -.. autosummary:: - :toctree: api/generated - :nosignatures: - :template: classtemplate.rst - - Accelerator - CPUAccelerator - CUDAAccelerator - MPSAccelerator - TPUAccelerator - - -Loggers -^^^^^^^ - -.. currentmodule:: lightning.fabric.loggers - -.. autosummary:: - :toctree: api/generated - :nosignatures: - :template: classtemplate.rst - - Logger - CSVLogger - TensorBoardLogger - -Precision -""""""""" - -.. TODO(fabric): include DeepSpeedPrecision - -.. currentmodule:: lightning.fabric.plugins.precision - -.. autosummary:: - :toctree: api/generated - :nosignatures: - :template: classtemplate.rst - - Precision - DoublePrecision - MixedPrecision - TPUPrecision - TPUBf16Precision - FSDPPrecision - - -Environments -"""""""""""" - -.. currentmodule:: lightning.fabric.plugins.environments - -.. autosummary:: - :toctree: api/generated - :nosignatures: - :template: classtemplate_noindex.rst - - ~cluster_environment.ClusterEnvironment - ~kubeflow.KubeflowEnvironment - ~lightning.LightningEnvironment - ~lsf.LSFEnvironment - ~mpi.MPIEnvironment - ~slurm.SLURMEnvironment - ~torchelastic.TorchElasticEnvironment - ~xla.XLAEnvironment - - -IO -"" - -.. currentmodule:: lightning.fabric.plugins.io - -.. autosummary:: - :toctree: api/generated - :nosignatures: - :template: classtemplate.rst - - ~checkpoint_io.CheckpointIO - ~torch_io.TorchCheckpointIO - ~xla.XLACheckpointIO - - -Collectives -""""""""""" - -.. currentmodule:: lightning.fabric.plugins.collectives - -.. autosummary:: - :toctree: api/generated - :nosignatures: - :template: classtemplate.rst - - Collective - TorchCollective - SingleDeviceCollective - - -Strategies -^^^^^^^^^^ - -.. TODO(fabric): include DeepSpeedStrategy, XLAStrategy - -.. currentmodule:: lightning.fabric.strategies - -.. autosummary:: - :toctree: api/generated - :nosignatures: - :template: classtemplate.rst - - Strategy - DDPStrategy - DataParallelStrategy - FSDPStrategy - ParallelStrategy - SingleDeviceStrategy - SingleTPUStrategy diff --git a/docs/source-fabric/glossary/index.rst b/docs/source-fabric/glossary/index.rst new file mode 100644 index 0000000000000..ec78566c4d04b --- /dev/null +++ b/docs/source-fabric/glossary/index.rst @@ -0,0 +1,180 @@ +######## +Glossary +######## + + +.. raw:: html + +
+
+ +.. displayitem:: + :header: Accelerator + :button_link: ../fundamentals/accelerators.html + :col_css: col-md-4 + +.. displayitem:: + :header: Apple Silicon + :button_link: ../fundamentals/accelerators.html + :col_css: col-md-4 + +.. displayitem:: + :header: Autocast + :button_link: ../fundamentals/precision.html + :col_css: col-md-4 + +.. displayitem:: + :header: Barrier + :button_link: ../advanced/distributed_communication.html + :col_css: col-md-4 + +.. displayitem:: + :header: Broadcast + :button_link: ../advanced/distributed_communication.html + :col_css: col-md-4 + +.. displayitem:: + :header: Callback + :button_link: ../guide/callbacks.html + :col_css: col-md-4 + +.. displayitem:: + :header: Checkpoint + :button_link: ../guide/checkpoints.html + :col_css: col-md-4 + +.. displayitem:: + :header: CLI + :button_link: ../fundamentals/launch.html + :col_css: col-md-4 + +.. displayitem:: + :header: Cloud + :button_link: ../guide/multi_node/cloud.html + :col_css: col-md-4 + +.. displayitem:: + :header: Collective + :button_link: ../advanced/distributed_communication.html + :col_css: col-md-4 + +.. displayitem:: + :header: CUDA + :button_link: ../fundamentals/accelerators.html + :col_css: col-md-4 + +.. displayitem:: + :header: Gather + :button_link: ../advanced/distributed_communication.html + :col_css: col-md-4 + +.. displayitem:: + :header: Gradient Accumulation + :button_link: ../advanced/gradient_accumulation.html + :col_css: col-md-4 + +.. displayitem:: + :header: GPU + :button_link: ../fundamentals/accelerators.html + :col_css: col-md-4 + +.. displayitem:: + :header: Jypyter + :button_link: ../launch/notebook.html + :col_css: col-md-4 + +.. displayitem:: + :header: Launch + :button_link: ../fundamentals/launch.html + :col_css: col-md-4 + +.. displayitem:: + :header: LightningModule + :button_link: ../guide/lightning_module.html + :col_css: col-md-4 + +.. displayitem:: + :header: Logger + :button_link: ../guide/logging.html + :col_css: col-md-4 + +.. displayitem:: + :header: Mixed Precision + :button_link: ../fundamentals/precision.html + :col_css: col-md-4 + +.. displayitem:: + :header: MPI + :button_link: ../guide/multi_node/other.html + :col_css: col-md-4 + +.. displayitem:: + :header: MPS + :button_link: ../fundamentals/accelerators.html + :col_css: col-md-4 + +.. displayitem:: + :header: Multi-GPU + :button_link: ../fundamentals/launch.html + :col_css: col-md-4 + +.. displayitem:: + :header: Multi-Node + :button_link: ../fundamentals/launch.html + :col_css: col-md-4 + +.. displayitem:: + :header: Notebook + :button_link: ../launch/notebook.html + :col_css: col-md-4 + +.. displayitem:: + :header: Optimizers + :button_link: ../advanced/multiple_setup.html + :col_css: col-md-4 + +.. displayitem:: + :header: Precision + :button_link: ../fundamentals/precision.html + :col_css: col-md-4 + +.. displayitem:: + :header: Reduce + :button_link: ../advanced/distributed_communication.html + :col_css: col-md-4 + +.. displayitem:: + :header: SLURM + :button_link: ../guide/multi_node/slurm.html + :col_css: col-md-4 + +.. displayitem:: + :header: TensorBoard + :button_link: ../guide/logging.html + :col_css: col-md-4 + +.. displayitem:: + :header: TorchElastic + :button_link: ../guide/multi_node/barebones.html + :col_css: col-md-4 + +.. displayitem:: + :header: TorchRun + :button_link: ../guide/multi_node/barebones.html + :col_css: col-md-4 + +.. displayitem:: + :header: TPU + :button_link: ../fundamentals/accelerators.html + :col_css: col-md-4 + +.. displayitem:: + :header: Trainer + :button_link: ../guide/trainer_template.html + :col_css: col-md-4 + + +.. raw:: html + +
+
diff --git a/docs/source-fabric/guide/index.rst b/docs/source-fabric/guide/index.rst new file mode 100644 index 0000000000000..bf1b97f1b4942 --- /dev/null +++ b/docs/source-fabric/guide/index.rst @@ -0,0 +1,163 @@ +############# +How-to Guides +############# + + +****** +Basics +****** + + +.. raw:: html + +
+
+ +.. displayitem:: + :header: Convert to Fabric in 5 minutes + :description: Learn how to add Fabric to your PyTorch code + :button_link: ../fundamentals/convert.html + :col_css: col-md-4 + :height: 150 + :tag: basic + +.. displayitem:: + :header: Scale your model with Accelerators + :description: Take advantage of your hardware with a switch of a flag + :button_link: ../fundamentals/accelerators.html + :col_css: col-md-4 + :height: 150 + :tag: basic + +.. displayitem:: + :header: Structure your Fabric code + :description: Best practices for setting up your training script with Fabric + :button_link: ../fundamentals/code_structure.html + :col_css: col-md-4 + :height: 150 + :tag: basic + +.. displayitem:: + :header: Launch distributed training + :description: Launch a Python script on multiple devices and machines + :button_link: ../fundamentals/launch.html + :col_css: col-md-4 + :height: 150 + :tag: basic + +.. displayitem:: + :header: Launch Fabric in a notebook + :description: Launch on multiple devices from within a Jupyter notebook + :button_link: ../fundamentals/notebooks.html + :col_css: col-md-4 + :height: 150 + :tag: basic + +.. displayitem:: + :header: Improve performance with Mixed-Precision training + :description: Save memory and speed up training using mixed precision + :button_link: ../fundamentals/precision.html + :col_css: col-md-4 + :height: 150 + :tag: basic + +.. raw:: html + +
+
+ + + +********************** +Build your own Trainer +********************** + +.. raw:: html + +
+
+ +.. displayitem:: + :header: Organize your model code with with LightningModule + :description: Organize your code in a LightningModule and use it with Fabric + :button_link: lightning_module.html + :col_css: col-md-4 + :height: 150 + :tag: intermediate + +.. displayitem:: + :header: Encapsulate code into Callbacks + :description: Make use of the Callback system in Fabric + :button_link: callbacks.html + :col_css: col-md-4 + :height: 150 + :tag: intermediate + +.. displayitem:: + :header: Track and visualize experiments + :description: Learn how Fabric helps you remove boilerplate code for tracking metrics with a logger + :button_link: logging.html + :col_css: col-md-4 + :height: 150 + :tag: intermediate + +.. displayitem:: + :header: Save and load model progress + :description: Efficient saving and loading of model weights, training state, hyperparameters and more. + :button_link: checkpoint.html + :col_css: col-md-4 + :height: 150 + :tag: intermediate + +.. displayitem:: + :header: Build your own Trainer + :description: Take our Fabric Trainer template and customize it for your needs + :button_link: https://github.com/Lightning-AI/lightning/tree/master/examples/fabric/build_your_own_trainer + :col_css: col-md-4 + :height: 150 + :tag: intermediate + +.. raw:: html + +
+
+ + +*************** +Advanced Topics +*************** + + +.. raw:: html + +
+
+ +.. displayitem:: + :header: Use efficient gradient accumulation + :description: Learn how to perform efficient gradient accumulation in distributed settings + :button_link: ../advanced/gradient_accumulation.html + :col_css: col-md-4 + :height: 160 + :tag: advanced + +.. displayitem:: + :header: Distribute communication + :description: Learn all about communication primitives for distributed operation. Gather, reduce, broadcast, etc. + :button_link: ../advanced/distributed_communication.html + :col_css: col-md-4 + :height: 160 + :tag: advanced + +.. displayitem:: + :header: Use multiple models and optimizers + :description: See how flexible Fabric is to work with multiple models and optimizers! + :button_link: ../advanced/multiple_setup.html + :col_css: col-md-4 + :height: 160 + :tag: advanced + +.. raw:: html + +
+
diff --git a/docs/source-fabric/index.rst b/docs/source-fabric/index.rst index cd4748f1c91c0..710146f2efb47 100644 --- a/docs/source-fabric/index.rst +++ b/docs/source-fabric/index.rst @@ -118,23 +118,34 @@ For alternative ways to install, read the :doc:`installation guide - Intermediate - Advanced + Basic skills + Intermediate skills + Advanced skills .. toctree:: :maxdepth: 1 - :name: api :caption: Core API Reference Fabric Arguments Fabric Methods - Utilities - Full API Reference + + +.. toctree:: + :maxdepth: 1 + :caption: Full API Reference + + Accelerators + Collectives + Environments + Fabric + IO + Loggers + Precision + Strategies .. toctree:: @@ -143,6 +154,9 @@ For alternative ways to install, read the :doc:`installation guide + Glossary + How-tos + Style Guide .. raw:: html diff --git a/docs/source-fabric/levels/advanced.rst b/docs/source-fabric/levels/advanced.rst index 49a735834e23d..25b4da65937cd 100644 --- a/docs/source-fabric/levels/advanced.rst +++ b/docs/source-fabric/levels/advanced.rst @@ -17,7 +17,7 @@ Advanced skills
.. displayitem:: - :header: Efficient Gradient Accumulation + :header: Use efficient gradient accumulation :description: Learn how to perform efficient gradient accumulation in distributed settings :button_link: ../advanced/gradient_accumulation.html :col_css: col-md-4 @@ -25,7 +25,7 @@ Advanced skills :tag: advanced .. displayitem:: - :header: Distributed Communication + :header: Distribute communication :description: Learn all about communication primitives for distributed operation. Gather, reduce, broadcast, etc. :button_link: ../advanced/distributed_communication.html :col_css: col-md-4 @@ -33,7 +33,7 @@ Advanced skills :tag: advanced .. displayitem:: - :header: Multiple Models and Optimizers + :header: Use multiple models and optimizers :description: See how flexible Fabric is to work with multiple models and optimizers! :button_link: ../advanced/multiple_setup.html :col_css: col-md-4 diff --git a/docs/source-fabric/levels/basic.rst b/docs/source-fabric/levels/basic.rst index cfc2296243506..cf53eff91d3d1 100644 --- a/docs/source-fabric/levels/basic.rst +++ b/docs/source-fabric/levels/basic.rst @@ -29,7 +29,7 @@ Basic skills :tag: basic .. displayitem:: - :header: Accelerators + :header: Scale your model with Accelerators :description: Take advantage of your hardware with a switch of a flag :button_link: ../fundamentals/accelerators.html :col_css: col-md-4 @@ -37,7 +37,7 @@ Basic skills :tag: basic .. displayitem:: - :header: Code Structure + :header: Structure your Fabric code :description: Best practices for setting up your training script with Fabric :button_link: ../fundamentals/code_structure.html :col_css: col-md-4 @@ -45,7 +45,7 @@ Basic skills :tag: basic .. displayitem:: - :header: Launch Distributed Training + :header: Launch distributed training :description: Launch a Python script on multiple devices and machines :button_link: ../fundamentals/launch.html :col_css: col-md-4 @@ -53,7 +53,7 @@ Basic skills :tag: basic .. displayitem:: - :header: Fabric in Notebooks + :header: Launch Fabric in a notebook :description: Launch on multiple devices from within a Jupyter notebook :button_link: ../fundamentals/notebooks.html :col_css: col-md-4 @@ -61,7 +61,7 @@ Basic skills :tag: basic .. displayitem:: - :header: Mixed Precision Training + :header: Improve performance with Mixed-Precision training :description: Save memory and speed up training using mixed precision :button_link: ../fundamentals/precision.html :col_css: col-md-4 diff --git a/docs/source-fabric/levels/intermediate.rst b/docs/source-fabric/levels/intermediate.rst index 85c7ef3d7d1d9..2e23022316257 100644 --- a/docs/source-fabric/levels/intermediate.rst +++ b/docs/source-fabric/levels/intermediate.rst @@ -19,7 +19,7 @@ Intermediate skills
.. displayitem:: - :header: The LightningModule + :header: Organize your model code with with LightningModule :description: Organize your code in a LightningModule and use it with Fabric :button_link: ../guide/lightning_module.html :col_css: col-md-4 @@ -27,7 +27,7 @@ Intermediate skills :tag: intermediate .. displayitem:: - :header: Callbacks + :header: Encapsulate code into Callbacks :description: Make use of the Callback system in Fabric :button_link: ../guide/callbacks.html :col_css: col-md-4 @@ -35,7 +35,7 @@ Intermediate skills :tag: intermediate .. displayitem:: - :header: Logging + :header: Track and visualize experiments :description: Learn how Fabric helps you remove boilerplate code for tracking metrics with a logger :button_link: ../guide/logging.html :col_css: col-md-4 @@ -43,7 +43,7 @@ Intermediate skills :tag: intermediate .. displayitem:: - :header: Checkpoints + :header: Save and load model progress :description: Efficient saving and loading of model weights, training state, hyperparameters and more. :button_link: ../guide/checkpoint.html :col_css: col-md-4 @@ -51,7 +51,7 @@ Intermediate skills :tag: intermediate .. displayitem:: - :header: Trainer Template + :header: Build your own Trainer :description: Take our Fabric Trainer template and customize it for your needs :button_link: https://github.com/Lightning-AI/lightning/tree/master/examples/fabric/build_your_own_trainer :col_css: col-md-4