From 16a782b81f159de795df50544dbdb2cabf58a4d0 Mon Sep 17 00:00:00 2001 From: Riccardo Musmeci Date: Thu, 14 Mar 2024 13:22:49 +0100 Subject: [PATCH] mlx-image (#1237) * add mlx-image * add mlx-image * fix for mlx-image * nit: blank line --------- Co-authored-by: Pedro Cuenca --- docs/hub/_toctree.yml | 2 + docs/hub/mlx-image.md | 98 +++++++++++++++++++++++++++++++++++++++++++ 2 files changed, 100 insertions(+) create mode 100644 docs/hub/mlx-image.md diff --git a/docs/hub/_toctree.yml b/docs/hub/_toctree.yml index 9344b3273..5475c0011 100644 --- a/docs/hub/_toctree.yml +++ b/docs/hub/_toctree.yml @@ -77,6 +77,8 @@ title: Keras - local: ml-agents title: ML-Agents + - local: mlx-image + title: mlxim - local: mlx title: MLX - local: open_clip diff --git a/docs/hub/mlx-image.md b/docs/hub/mlx-image.md new file mode 100644 index 000000000..f20c022a5 --- /dev/null +++ b/docs/hub/mlx-image.md @@ -0,0 +1,98 @@ +# Using mlxim at Hugging Face + +[`mlxim`](https://github.com/riccardomusmeci/mlx-image) is an image models library built on Apple [MLX](https://github.com/ml-explore/mlx). It tries to replicate the great [timm](https://github.com/huggingface/pytorch-image-models) library from Ross Wightman, but for MLX models. + + +## Exploring MLX on the Hub + +You can find `mlxim` models by filtering using the `mlxim` library name, like in [this query](https://huggingface.co/models?library=mlxim&sort=trending). +There's also an open [mlx-vision](https://huggingface.co/mlx-vision) space for contributors converting and publishing weights for MLX format. + +Thanks to MLX Hugging Face Hub integration, you can load MLX models with a few lines of code. + +## Installation + +```bash +pip install mlx-image +``` + +## Models + +Model weights are available on the [`mlx-vision`](https://huggingface.co/mlx-vision) community on HuggingFace. + +To load a model with pre-trained weights: +```python +from mlxim.model import create_model + +# loading weights from HuggingFace (https://huggingface.co/mlx-vision/resnet18-mlxim) +model = create_model("resnet18") # pretrained weights loaded from HF + +# loading weights from local file +model = create_model("resnet18", weights="path/to/resnet18/model.safetensors") +``` + +To list all available models: + +```python +from mlxim.model import list_models +list_models() +``` +> [!WARNING] +> As of today (2024-03-08) mlx does not support `group` param for nn.Conv2d. Therefore, architectures such as `resnext`, `regnet` or `efficientnet` are not yet supported in `mlxim`. + +## ImageNet-1K Results + +Go to [results-imagenet-1k.csv](https://github.com/riccardomusmeci/mlx-image/blob/main/results/results-imagenet-1k.csv) to check every model converted to `mlxim` and its performance on ImageNet-1K with different settings. + +> **TL;DR** performance is comparable to the original models from PyTorch implementations. + + +## Similarity to PyTorch and other familiar tools + +`mlxim` tries to be as close as possible to PyTorch: +- `DataLoader` -> you can define your own `collate_fn` and also use `num_workers` to speed up data loading +- `Dataset` -> `mlxim` already supports `LabelFolderDataset` (the good and old PyTorch `ImageFolder`) and `FolderDataset` (a generic folder with images in it) +- `ModelCheckpoint` -> keeps track of the best model and saves it to disk (similar to PyTorchLightning). It also suggests early stopping + +## Training + +Training is similar to PyTorch. Here's an example of how to train a model: + +```python +import mlx.nn as nn +import mlx.optimizers as optim +from mlxim.model import create_model +from mlxim.data import LabelFolderDataset, DataLoader + +train_dataset = LabelFolderDataset( + root_dir="path/to/train", + class_map={0: "class_0", 1: "class_1", 2: ["class_2", "class_3"]} +) +train_loader = DataLoader( + dataset=train_dataset, + batch_size=32, + shuffle=True, + num_workers=4 +) +model = create_model("resnet18") # pretrained weights loaded from HF +optimizer = optim.Adam(learning_rate=1e-3) + +def train_step(model, inputs, targets): + logits = model(inputs) + loss = mx.mean(nn.losses.cross_entropy(logits, target)) + return loss + +model.train() +for epoch in range(10): + for batch in train_loader: + x, target = batch + train_step_fn = nn.value_and_grad(model, train_step) + loss, grads = train_step_fn(x, target) + optimizer.update(model, grads) + mx.eval(model.state, optimizer.state) +``` + +## Additional Resources + +* [mlxim repository](https://github.com/riccardomusmeci/mlx-image) +* [All mlxim models on Hub](https://huggingface.co/models?library=mlxim&sort=trending)