This chapter is one person's opinionated overview of the ML/AI Engineering reality, which may or may not be another person's reality. The intention is to help you start asking the right questions and get your ML Engineering needs met.
Training:
- How fast one can train a better model (first to market advantage)
- How much $$ was spent (do we still have money left to pay salaries to talent after training?)
Inference:
- Fast latency (users are used to msec response times and will leave if the response takes seconds)
- Fast throughput (how many concurrent queries can be processed)
- How much $$ is being spent per user (can we rent more GPUs to acquire more users and/or improve (1) and (2)?)
- Fast compute massively dominated by matrix multiplications
- Fast enough memory, IO, network and CPU to feed the compute
Corollary: If when you buy or rent hardware you invest in the fastest accelerators, but cheap out on any of the other components you wasted $$ and you might not win the race as it'll take longer to train.
-
An accelerator or a processing unit is what does most of the work.
-
Since ML does a lot of parallel processing (SIMD) GPUs were used at the beginning, but now you additionally have TPUs, IPUs, FPGAs, HPUs, QPUs, RDUs, etc. Recent CPUs are becoming used as accelerators as well, especially for inference.
- AI companies - train models/build products around self-trained or trained-by-others' models, in-house research.
- Academia - does massive research and write papers. Lots of new ideas are generated.
- AI enthusiasts - lots of good will available, some pull resources/talents together to train open access models, with donated compute by HPCs and an occasional cloud, or a university cluster.
- Entrepreneurs - lots of low hanging fruit to pick - creative reselling of services, making ML-driven apps, and using various ingenious combinations of available resources to create amazing outcomes.
- It's very surprising that almost everybody involved in the domain of AI shares a lot of the discoveries with the community.
- Surely, companies don't disclose all of their IP, but a lot of it does get shared in the form of knowledge or model weights
- Companies that publish a lot of IP and models tend to attract higher quality talent.
- Twitter seems to be the central platform where one must be to follow what's going on
-
The Dot-com bubble occurred during 1995-2000. And a very similar situation is happening right now in the AI space.
-
There is a lot of money available to create new startups or boost the existing companies. It's relatively easy to raise millions of dollars.
-
As we are in the wild-wild-west stage of the AI industry it's very difficult to predict the future, and so pretty much anything goes as far as startup ideas go, as long as it sounds reasonable.
-
What distinguishes the AI bubble from the Dot-com bubble, is that one didn't actually need much money to operate a Dot-com company - most of the raised money went to marketing and some to staff, barely any to compute. AI companies need millions of dollars because training LLMs requires an insane amount of compute, and that compute is very expensive. e.g. 1x NVIDIA H100 costs ~$30k and a company may need 512 of those, which is $15M (not counting the other hardware components and related costs)!
This is my personal LLM/VLM trainings-based heaven and hell. YMMV.
-
A well built HPC, or a full service cloud based cluster, where someone diligently and timely takes care of the hardware and the systems.
I just need to bring my training software and do the training, which is already an insanely complicated job requiring special skills.
-
Lots of nodes available for exclusive unlimited use
-
Fast inter-node connectivity that doesn't bottleneck the accelerators and which isn't shared with other users
-
Huge local super-fast NVME based shared filesystem that can fit datasets and checkpoints
-
Barebones Linux w/ SLURM and minimal software to be able to launch training jobs
-
sudo
er access to ease the work with a team of people
-
A cloud or in-house cluster, where you have to do everything - sysadmining, replacing hardware, dealing with outages, etc. And to do the training on top of that.
-
A smallish slow shared filesystem (NFS?), with cloud to draw data from and checkpoint to
-
Slow inter-node leading to low accelerator utilization
-
Inter-node shared with other users which make the network erratic and unpredictable
-
Super-complicated cloud console with gazillion of screens and steps to set even simple things up
-
Not being able to swap out failing hardware fast
-
Needing to timeshare the nodes - with wait times between training jobs
-
Having other concurrent users who might use up the whole disk, leading to trainings crashing
-
Not being able to kill jobs others on the team started and went to sleep
There are 3 main choices to where one gets compute:
- Rent on the cloud
- Get a timeshare on an HPC
- Buy it
This is currently the prevalent way of getting compute.
Pros:
- Easy to expand or contract the size of the cluster
- Easy to upgrade from the old hardware generation to the new one in a few years
- Cluster management could be easily outsourced
Cons:
- Expensive, unless you negotiate a long term (1-3 year) contract for hundreds of accelerators
- You will be tempted to buy many tools and services that you may or may not need
- You always get charged whether you use your cluster fully or not
There aren't that many HPCs out there and so the amount of available resources is limited.
Pros:
- Managed for you - all you need is your software to do the training and a bit of SLURM know-how to launch jobs
- Often sponsored by the local government/university - probably could get the job done for less $$ or even free (e.g. we trained BLOOM-176B for free on JeanZay HPC!)
Cons:
- needing to time share compute with other teams == short job times with possible long wait times in between - could be difficult to finish training quickly
- The inter-node network is likely to be unstable as it'll be used by other teams
- Have to abide by the HPC's rules (e.g. no
sudo
access and various other rules to follow) - In a way the HPC cluster will be what it'll be - you can't make the network faster and often even getting some software installed can be tricky.
It's mainly universities that buy and build their own clusters, and some big companies do that too.
Pros:
- If you can deploy the hardware 24/7 for more than a few years the total cost will be cheaper than renting
- Easy to provide fast local storage - a good NVME raid would be much cheaper and faster than online storage
Cons:
- You're stuck with the outdated hardware just a few years after it was purchased - might be able to resell
- Must buy more than needed - Hardware tends to break, especially when it's used 24/7, RMA could take weeks
- Have to hire talent to manage the in-house solution
- Have to figure out cooling, electric costs, insurance, etc.
- Unless you use a fully managed HPC compute you absolutely need to hire a sysadmin. It may feel that your ML engineers can swing that between their training jobs, but they will be losing a lot of time to managing disk space, dealing with problematic nodes, asking users to behave, etc.
Imagine a steam locomotive - the engine is great, but if the fireman isn't fast enough to shovel the coal in, the train won't move fast.
This is the current state of ML hardware: The bottleneck is in moving bits and not the compute.
- Accelerators get ~2x faster every 2 years (Moore's law)
- Network and memory are not! Already now both are compute bottlenecks
- IO can be another bottleneck if your DataLoader has to pull data from the cloud
- CPU is fine as long as it has enough cpu-cores for DataLoader workers, and main processes
Corollary: research the whole machine and not just its engine.
a crazy idea: the older GPUs might do fine if you can actually feed them as fast as they can compute. And if you can get 3x of them at the same cost as the next generation GPU you might finish training sooner and a lower cost.
-
Once you choose the architecture and the size of the model and how many tokens you want to train the model for you immediately know how much compute will be required to accomplish this goal. Specifically you can now calculate how many floating point operations will be needed.
-
All that is missing is comparing different compute providers to how many floating point operations their hardware can computes per secs (TFLOPS) and their cost per unit and now you can tell the total approximate cost of the training.
-
Calculate the time needed to train given the TFLOPS of the considered solution:
total_tflops_required / tflops_of_this_compute_unit = time_in_seconds
Let's say it came to be 604800 secs or 7 days.
-
Look at the cost of using this compute solution for 7 days and now you know the total $$ to train this model.
-
Look at other proposals and calculate the same - chose the best option.
- As mentioned earlier, time is of a huge importance, so you might still choose a more expensive solution if finishing the training sooner is important because you want to be first to market.
Unfortunately, this math is only partially correct because the advertised peak TFLOPS are typically unachievable. The MFU section delves into it.
As mentioned in the previous section, some (most?) vendors publish unrealistic peak performance TFLOPS - they aren't possible to achieve.
Model Flops Utilization (MFU) is the metric that tells us how well the accelerator is utilized. Here is how it is calculated:
- Measure the actual TFLOPS by calculating how many floating point operations a single training iteration takes and dividing that number by the number of seconds this iteration took.
- Divide the actual TFLOPS by advertised TFLOPS to get the MFU
Example: Let's say you're training in BFLOAT16 precision:
- If a single iteration requires 624 Tera floating point operations and it took 4 secs to run then we know that we get:
624/4=156
actual TFLOPS - now BF16@A100 is advertised as 312TFLOPS so
156/312=0.5
gives us 50% MFU.
Practically:
- with NVIDIA GPUs if you're above 50% MFU on a multi-node setup with a large model you're already doing fantastic
- recent advancements in more efficient scalability solutions keep on increasing MFU
- slow networks and inefficient frameworks or untuned configuration lower MFU
Therefore once you know the MFU you can now adjust the cost estimate from the previous section. In the example there we said it'll take 7 days to train, but if MFU is 50%, it means it'll take 14 days to train.
Why can't the advertised TFLOPS achieved? It's because it takes time to move data between accelerator memory and compute and additionally it takes even more time to move data from disk and other gpus to the accelerator's memory.
-
There is not much can be done about the accelerator memory since its bandwidth is what it is - one can only write more efficient software to make data move faster to/from the accelerator - hint: fused and custom written kernels (like torch.compile and flash attention)
-
If you only have a single GPU and the model fits its memory, you don't need to worry about the network - accelerator memory is the only bottleneck. But if you have to shard the model across multiple GPUs network becomes the bottleneck.
-
Intra-node Network - is very fast, but difficult to take advantage of for large models - Tensor parallelism and sequence parallelism address part of this problem. (more).
-
Inter-node Network - typically is too slow on most server setups - thus this is the key component to research! Efficient frameworks succeed to partially hide the comms overhead by overlapping compute and comms. But if comms take longer than compute, the comms are still the bottleneck. more.
-
Storage IO is important primarily for feeding the DataLoader workers and saving the checkpoints. more.
- Typically with enough DL workers the DataLoader adds very little overhead.
- While checkpoints are being saved the accelerators idle unless some async saving solution is used, so fast IO is crucial here
As of this writing here are the most common accelerators that can be used for training, finetuning and inference ML models:
Widely available:
- NVIDIA A100 - huge availability across all clouds, but is already gradually being replaced by H100
Available, but locks you in:
- Google TPUs - fast! but the cost is a lock-in into a single vendor and cloud
Emerging to general availability:
-
NVIDIA H100 - 2-3x faster than A100 (half precision), 6x faster for fp8
-
AMD MI250 ~= A100 - very few clouds have them and most likely MI300X will be the first mainstream AMD GPU
-
AMD MI300X ~= H100 - a few clouds will have those in March, 2024
-
Intel Gaudi2 ~= H100 - starting to slowly emerge on Intel's cloud
-
GraphCore IPU - very difficult to find, paperspace has them
-
Cerebras WaferScale Engine - available on Cerebras' cloud
For the full list see Accelerators.
In general most (all?) accelerators are supported by major frameworks like PyTorch or TensorFlow and the same code should run everywhere with small modifications as long as it doesn't use any accelerator-specific functionality.
For example, if your PyTorch application calls torch.mm
- it should work everywhere, but if it includes custom CUDA kernels it'll only work on NVIDIA GPUs and may be on the recent AMD MI-series.
-
NVIDIA GPUs: all based on CUDA, which most training frameworks support. You can easily moved between different NVIDIA GPUs and most things would work the same.
-
AMD MI250/MI300X: with PyTorch using ROCm you can run most CUDA-based software as is. This is really the only inter-operable accelerator with the NVIDIA stack.
-
Gaudi2: if you use HF Transformers/Diffusers you can use optimum-habana. If you use HF Trainer with NVIDIA GPUs it should be relatively easy to switch to train/infer on Gaudi2.
-
GraphCore IPU: can also be run via PyTorch via poptorch
-
Cerebras: is also working on PyTorch support via Cerebras Software Platform (CSoft) via XLA.
Also in general most ML code could be compiled into cross-platform formats like Open Neural Network Exchange (ONNX) which can be run on a variety of accelerators. This approach is typically used more often for inference workloads.
-
If you want to train a large model that doesn't fit onto a single accelerator's memory you have to rely on the intra- and inter-node networks to synchronize multiple accelerators.
-
The biggest issue right now is that compute hardware advancements move faster than networking hardware, e.g. for NVIDIA NVLink intra-node:
GPU | Computefp16 TFLOPS |
Compute speedup |
Intra-node GBps |
Intra-node speedup |
---|---|---|---|---|
V100 | 125 | 1 | 300 | 1 |
A100 | 312 | 2.5 | 600 | 2 |
H100 | 989 | 8 | 900 | 3 |
-
You can see that A100 was 2.5 faster than V100, and H100 is ~3x faster than A100. But the intra-node speed of NVLink has only increased by 300GBps each generation.
-
Moreover, all 3 generations of NVLink use identical NICs of the same 50GBps duplex throughput. They have just doubled and tripled the number of links to speed things up. So there was 0 progress in that technology.
-
The inter-node situation isn't any better with most NICs there doing 100 or 200Gbps, and some 400Gbps are starting to emerge. (correspondingly in GBps: 12.5, 25 and 50). It's the same story here, some solutions provide dozens of NICs to get to higher speeds.
-
Also typically with LLMs the payload is so large that network latency is often negligible for training. It's still quite important for inference.
-
Pay attention to bytes vs bits. 1Byte = 8bits. 1GBps = 8Gbps.
-
If you need to reduce bits (e.g. gradients) across multiple nodes, it's the slowest link (Inter-node) that defines the overall throughput, so intra-node speed doesn't matter then
-
Tensor parallelism and sequence parallelism have to remain within the node to be efficient - only makes sense with fast intra-node speed
NVIDIA:
-
NVIDIA-based compute nodes come with 50GBps duplex NVLInk
-
Some have a lot of NVLinks, others less but typically plenty w/ at least 900GBps (5.6Tbps) duplex for H100, 600GBps for A100 nodes
Intel Gaudi2:
- 8x 21 NICs of 100GbE RoCE v2 ROMA for a total of 2.1TBps
-
An order of magnitude slower than Intra-node
-
You will see a wide range of speeds from 50Gbps to 3200 Gbps
-
You need to reduce gradients and other bits faster than compute to avoid idling accelerators
-
You typically get at most 80% of advertised speed. e.g., if you are told you get 800Gbps, expect ~640Gbps.
-
If moving to fp8 H100 is 18x faster than V100
-
We are yet to see if 3200Gbps for H100s will be enough to keep high MFU.
- Practically less than 3x but it's a good estimate
There are 3 distinct Storage IO needs in the ML workload:
- You need to be able to feed the DataLoader fast - (super fast read, don't care about fast write) - requires sustainable load for hours and days
- You need to be able to write checkpoints fast - (super fast write, fastish read as you will be resuming a few times) - requires burst writing - you want super fast to not block the training for long (unless you use some sort of cpu offloading to quickly unblock the training)
- You need to be able to load and maintain your codebase - (medium speed for both reading and writing) - this also needs to be shared since you want all nodes to see the same codebase - as it happens only during the start or resume it'll happen infrequently
-
Most of the time you're being sold 80% of what you paid. If you want a reliable 100TBs you need to rent 125TBs or your application may fail to write long before the disk is full.
-
Shared Distributed Filesystem:
- non-parallel shared file systems can be extremely slow if you have a lot of small files (=Python!)
- You want Parallel FS like GPFS (IBM Spectrum Scale) or Lustre (Open Source)
You need enough memory for:
-
2-3 possibly DL workers per Accelerator (so 16-24 processes with 8 accelerators per node)
-
Even more memory for DL workers if you pull data from the cloud
-
Enough memory to load the model if you can't load to accelerator directly
-
Often used for accelerator memory offloading - extends accelerator's memory by swapping out the currently unused layers - if that's the target use, then the more cpu memory is available - the better!
This is probably the least worrisome component.
-
Most clouds provide beefy CPUs with plenty of cpu cores
-
You need to have enough cores to run 2-3 DL workers +1 per gpu - so at least 30 cores
-
Even more cores for DL workers if you have complex and/or slow DL transforms (CV)
-
Most of the compute happens on GPUs
-
Training in half mixed-precision:
model_size_in_B * 18 * 1.25 / gpu_size_in_GB
-
Inference in half precision:
model_size_in_B * 2 * 1.25 / gpu_size_in_GB
That's the minimum, more to have a bigger batch size and longer sequence length.
Here is the breakdown:
-
Training: 8 bytes for AdamW states, 4 bytes for grads, 4+2 bytes for weights
-
Inference: 2 bytes for weights (1 byte if you use quantization)
-
1.25 is 25% for activations (very very approximate)
For example: Let's take an 80B param model and 80GB GPUs and calculate how many of them we will need for:
- Training: at least 23 GPUs
80*18*1.25/80
- Inference: at least 3 GPUs
80*2*1.25/80
As you navigate this very complex AI industry here are some thing to be aware of:
-
If you contract doesn't have clear deliverables (time and performance) don't be surprised if you paid for something you won't receive in time you need it or not at all
-
Be very careful before you sign a contract that includes clauses that start with "we will make a reasonable effort to ...".
When was the last time you went to the bread section of the supermarket and found a lump of half-baked dough with a note "we made a reasonable effort to bake this bread, but alas, what you see is what you get"?
But for whatever reason it's acceptable to create a legal contract where the provider provides neither delivery dates nor performance metrics and doesn't provide stipulations for what will they do in recompense when those promises aren't fulfilled.
-
Some cloud providers will make you use very proprietary tools or hardware that will make it very difficult for you to leave down the road because you will have to retool everything if you leave
-
Consider what would be the cost of moving to a different provider should this provider prove to be not satisfactory or if they don't have a capacity to fulfill your growing needs.
-
If you rent a cluster with a generic Linux box with generic open source tools it should be trivial to move from one provider to another as almost everything would work out of the box
-
Obviously if you choose compute that requires custom software that works for that hardware only and you can't rent this hardware anywhere else you're setting yourself up for a lock-in
-
The cloud providers have mostly the same generic hardware, which leads to a very slim $$ margin and so in order to make big $$ they invent products and then try to convince you that you need to buy them. Sometimes you actually need those products, but very often not. See also the previous section on lock-in, since proprietary products usually mean a partial lock-in.
-
Often it's easy to observe the 3 step marketing technique for solutions that seek a problem to solve:
- Convince a couple of well respected customers to use the provider's proprietary products by giving them huge discounts or even pay them to use them
- Use those in step 1 as the social approval lever to reel in more converts
- Then scoop the rest of the strugglers by telling them that 80% of your customers (1+2) use these amazing products
When marketing these products it's important:
- to mention how well they work with a dozen of other products, since now you're not buying into a single product but into a whole proprietary product-sphere.
- to use really nice looking complicated diagrams of how things plug into each other, and move really fast to the next slide before someone asks a difficult question.
HPCs are probably a good group of compute providers to learn from - they have no funds to create new products and so they creatively address all their needs using mostly generic open source tools with some custom written software added when absolutely needed.
To conclude I thought I'd share some insights to how one could slightly improve their daily AI battlefield experience.
If you read Twitter and other similar ML-related feeds you're guaranteed to feel the fear of missing out, since there is probably at least one new great model getting released weekly and multiple papers are getting published daily and your peers will publish their cool achievements hours.
We are dealing with very complicated technology and there is a small handful of people who can absorb that much new material and understand / integrate it.
This can be extremely depressing and discouraging.
I deal with it by looking at twitter about once or twice a week. I mostly use Twitter in broadcast mode - that is if I have something to share I post it and only watch for possible follow up questions.
Usually all the important news reach me through other people.
The pace of innovation in the field of AI is insane. It's not possible to know all-things-AI. I'd dare to say it's not possible to know even 10% of it for most of us.
I realized this very early one and I stopped paying attention to most announcements, tutorials, keynotes, etc. Whenever I have a new need I research it and I discover what I need and I have to be careful not to try to learn other things not pertinent to the goal at hand.
So I actually know very little, but what I have researched in depth I know quite well for some time and later I forget even that (that's why I write these notes - so that I can easily find what I have already researched).
So if you ask me something, chances are that I don't know it, but the saving grace for me is that if you give me time I can figure it out and give the answer or develop a solution.
Because the ML field is in a huge race, a lot of the open source software is half-baked, badly documented, badly tested, at times poorly supported. So if you think you can save time by re-using software written by others expect spending hours to weeks trying to figure out how to make it work. And then keeping it working when the updates break it.
The next problem is that most of this software depends on other software which often can be just as bad. It's not uncommon where I start fixing some integration problem, just to discover a problem in a dependent package, which in its turn has another problem from another package. This can be extremely frustrating and discouraging. Once excepts to save time by reuse, but ends up spending a long time figuring out how to make it work. At least if I write my own software I have fun and it's a creative process, trying to make other people's software work is not.
So at the end of the day we are still better off re-using other people's software, except it comes at an emotional price and exhaustion.
So first of all, try to find a way not to beat yourself up if the software you didn't write doesn't work. If you think about it, those problems aren't of your creation.
Learning how to debug efficiently should also make this process much less painful.