Before the transformers architecture was developed, which was done by Google in 2017. Recursive Neural Networks (RNN) had been used up until this point but transformers outperformed them in terms of translation quality and training cost.
One important thing to note about transformers is that if you have an input sequence lenght of N elements, you end up making N×N comparisons, which is N² (quadratic scaling). This means the computational cost grows very quickly as the length of the sequence increases.
Transformers are a type of neural network architecture that is used for processing sequential data. They are used for tasks like translation, text generation, and summarization. So they can take text as input but neural networks don't process text, instead the text need to be transformed into a format that the neural network can work with. This is done by first tokenizing the text, and then these tokens are converted into [embeddings] which are vectors of numbers that represent the tokens. These embeddings are then fed into the neural network.
One thing to keep in mind when looking at the "diagram" below is that it is used while training a model and also when the model is used for inference. So during training the weights and biases are updated and the finished result of training is a model that can be used for inference (where the weights are just loaded and not updated).
This destinction is particularly important when looking at the decoder as the "input" to the decoder during training might be the target sequence (for example when doing translation would be the translated sentence in the target language) , but during inference the input will be the start-of-sequence-token and this sequence grows with each step as the model generates a new word.
[Encoder]
+-----------------------------+
| Input Sequence |
| "Dan loves icecream" |
+-----------------------------+
|
↓
+-----------------------------+
| Tokenization & Embedding |
+-----------------------------+
|
↓
+-----------------------------+
| Positional Encoding |
+-----------------------------+
|
↓
+------------------------------+
| Encoder Layer 1 |
| | |
|+-----------| |
|| | |
|| +---------+---------+ |
|| |Q |K |V |
|| +-------------------------+ |
|| | Multi-Head Attention | |
|| +-------------------------+ |
|| | |
|+--+ | |
| ↓ ↓ |
| +-------------------------+ |
| | Add & Norm | |
| +-------------------------+ |
| | |
|+------------| |
|| | |
|| ↓ |
|| +-------------------------+ |
|| | Feed-Forward Network | |
|| +-------------------------+ |
|| | |
|+--+ | |
| ↓ ↓ |
| +-------------------------+ |
| | Add & Norm | |
| +-------------------------+ |
| |Q |K |
| | | |
+---|---------|----------------+
| +---------------------+
+-----------------------------+ |
| |
[Decoder] | |
| |
+-----------------------------+ | |
| Outputs Sequence | | |
| | | |
| Training: "Dan älskar glass"| | |
| Inference: "<sos>" | | | sos = start of sequence token
+-----------------------------+ | |
| | |
↓ | |
+-----------------------------+ | |
| Tokenization & Embedding | | |
+-----------------------------+ | |
| | |
↓ | |
+-----------------------------+ | |
| Positional Encoding | | |
+-----------------------------+ | |
| | |
↓ | |
+------------------------------+ | |
| Decoder Layer 1 | | |
| | | | |
|+-----------| | | |
|| | | | |
|| +---------+---------+ | | |
|| |Q |K |V | | |
|| +-------------------------+ | | |
|| | Masked Multi-Head | | | |
|| | Attention | | | |
|| +-------------------------+ | | |
|| | | | |
|+--+ | | | |
| ↓ ↓ | | |
| +-------------------------+ | | |
| | Add&Norm | | | |
| +-------------------------+ | | |
| | +-------------------+ |
|+---| | +-----------+
|| | | | |
|| ↓V ↓Q ↓ K |
|| +------------------------+ |
|| | Multi-Head Attention | |
|| +------------------------+ |
|| | |
|+--+ | |
| ↓ ↓ |
| +-------------------------+ |
| | Add&Norm | |
| +-------------------------+ |
| | |
|+------------| |
|| | |
|| ↓ |
|| +-------------------------+ |
|| | Feed-Forward Network | |
|| +-------------------------+ |
|| | |
|+--+ | |
| ↓ ↓ |
| +-------------------------+ |
| | Add&Norm | |
| +-------------------------+ |
| | |
| ↓ |
| +-------------------------+ |
| | Linear layer | |
| +-------------------------+ |
| | |
| ↓ |
| +-------------------------+ |
| | Soft max | |
| +-------------------------+ |
+------------------------------+
Encoders are used for classification and regression tasks. Decoders are used for text generation tasks, like translation and summarization.
Encoder-Decoder are used for task like generative text like translation or summarization. So if we wanted to train a model to translate a sentence from English to Swedish, we would have the English sentence as input and the Swedish sentence as the output. The model would then be trained to predict the Swedish sentence given the English sentence.
Now, lets take a closer look at the boxes of the above diagram. I've written about embeddings in embeddings.md. and positional encoding in positional-encoding.md so lets skip them for now and start with the encoder layer. So the next layer, or most often multiple layers, is the multi-head attention.
Standard attention uses 3 martixes, a query matrix, a key matrix, and a value matrix.
Let's start with the following input sentence "Dan loves icecream". The first step is to split this into tokens, so we will have might get 4 tokens:
["Dan", "loves", "ice", "cream"]
Next, these words are mapped to token id from the model's vocabulary:
[1003] [82] [371] [10004]
Now, the model will the take these input and map them into embeddings which might be of a dimension of say 512. So we will have 4 vectors of 512 dimensions
'Dan' 1003 [0 ... 512]
'loves' 82 [0 ... 512]
'ice' 371 [0 ... 512]
'cream' 10004 [0 ... 512]
If the same word appears multiple times in the input, the same embedding will be used for each occurance. So there is currently no context or association between these words/token embeddings. They only contain information about each word/token itself, and nothing about the context in which it appears.
This mapping can happen using somethin like ggml_get_rows
which uses the a
tensor that contains the embeddings for each token, and an index tensor which
contains the token ids. The index tensor is used to index into the embeddings.
So with these embeddings the first thing in the model does is to add a positional encoding to each of the embeddings. In the original paper this used absolute position encoding. I've written about this is embeddings.md.
So we have our input matrix which in our case is a 4x512 matrix, where each entry is one of the tokens in the input sentence. Notice that we in this case have a sequence lenght of 4 (tokens that is). If we had a longer sequence this work increase the size of the matrix. This has implications with regards to memory usage and computation when the sequence lenghts get longer.
We take this matrix and make four copies, but we will only focus on the first three for now:
+--------+ +-------+
| Input | -------> | Query |
| | +-------+
+--------+ -------> +-------+
| Key |
+-------+
+-------+
-------> | Value |
+-------+
+--------------+
--------------------------------> | Feed forward |
+--------------+
The attention function looks like this:
Attention(Q, K, V) = softmax((Q x Kᵗ)/√embedding_dim) x V
embedding_dim = 512 in our example
Lets start with looking at the nominator which is the matrix multiplication of Q and K transposed:
+----------------+ +-------+ +--------+
| Q | | Key | | |
| | X | | = | |
| (4, 512) | |(512,4)| | (4, 4) |
+----------------+ | | +--------+
| |
| |
+-------+
So, lets just think about this for a second. We know we copied the input matrix to Q and K. So I think we can visualize the attention score matrix like this:
(A) (B) (C) (D)
+-------------------+
Dan (A) |AA | AB | AC | AD |
|-------------------|
loves (B) |BA | BB | BC | BD |
|-------------------|
ice (C) |CA | CB | CC | CD |
|-------------------|
cream (D) |DA | DB | DC | DD |
+-------------------+
So AB is the dot product of the embeddings for word "Dan" and the embeddings for the word "loves". Notice how we are "comparing" the word "Dan" with all the other words in the sentence. And we do this for all words/tokens as well. The dot product will give us some value that indicates how similar the two words are (how far apart they are in the embedding space). Once again I'd like to point out that if we had a longer sequence the QK matrix would be larger.
The next thing we do is we scale the values in the matrix by dividing them by
the square root of the embedding dimension. Recall that this called the
scaled dot product attention
and this is the scaling part. This is done to
avoid stability issues if the dot product values are too large.
So we divide each value in the matrix with the square root of the embedding dimension. After that we apply the softmax function to the matrix. This will give us a matrix where the values are between 0 and 1 and the sum of the values in each row will be 1. This matrix tells us how much attention we should pay to each word in the input sentence. We take this matrix and multiply it with the V matrix which is just the input matrix unchanged. The resulting matrix will give us the attention scores for each of the words/tokens.
I just want to point out that this was using a single Q, K, and V matrix which
could be called single head attention
. And also notice that there were no
learnable parameters in this case. We only used the input matrix which was
copied into Q, K, and V. In actual implementation what is used is something
called multi-head attention
which I'll try to explain now.
So we have our input matrix like before and we create copies of it just the same as for single head attention:
+--------+ +-------+
| Input | -------> | Query |
| | +-------+
+--------+ -------> +-------+
| Key |
+-------+
+-------+
-------> | Value |
+-------+
+--------------+
--------------------------------> | Feed forward |
+--------------+
But we also have additional matrices which are learnable, meaning they can be updated by the model during training.
+--------+ +-------+ +-------+ +-------+
| Input | -------> | Query | X | W^q | = | Q' |
| | +-------+ +-------+ +-------+
+--------+ -------> +-------+ +-------+ +-------+
| Key | X | W^k | = | K' |
+-------+ +-------+ +-------+
+-------+ +-------+ +-------+
-------> | Value | X | W^v | = | V' |
+-------+ +-------+ +-------+
Query = (seq_len, embd_dim)
Key = (seq_len, embd_dim)
Value = (seq_len, embd_dim)
W^q = (embd_dim, embd_dim)
W^k = (embd_dim, embd_dim)
W^v = (embd_dim, embd_dim)
Q' = (seq_len, embd_dim)
K' = (seq_len, embd_dim)
V' = (seq_len, embd_dim)
So the above is just a single head attention but this would be the same thing if we have a multihead attention but the number of heads is 1.
The multi-head attention
function looks like this:
MultiHeadAttention(Q, K, V) = Concat(head_1, ..., head_h) x W^o
head_i = Attention(QW^qᵢ, KW^kᵢ, VW^Vᵢ)
Attention(Q, K, V) = softmax(Q, K, V) x V
h = number of heads
d_model = embedding dimension size
dₖ = d_model / h For example 4 heads and d_model = 512, dₖ = 128
If we look at the Attention function it is the same as we saw earlier. What is new is that we are going to split the matrices Q', K', and V' into smaller matrices. This is the number of heads that we have.
So for example, if we want to have 4 heads and the embedding dimension size is 512, then we will have 4 4x128 matrices. Each one of these are called a head and they are separate from each other, and are used to perform the single-head attention function that we went through above.
+--------+ +-------+ +-------+ +-------+
| Input | -------> | Query | X | W^q | = | Q' |
| | +-------+ +-------+ +-------+
+--------+ -------> +-------+ +-------+ +-------+
| Key | X | W^k | = | K' |
+-------+ +-------+ +-------+
+-------+ +-------+ +-------+
-------> | Value | X | W^v | = | V' |
+-------+ +-------+ +-------+
seq_len = 4
embd_dim = 512
heads = 1
Query = (4, 512)
Key = (4, 512)
Value = (4, 512)
W^q = (512, 512)
W^k = (512, 512)
W^v = (512, 512)
Q' = (4, 512)
K' = (4, 512)
V' = (4, 512)
So that would produce three matrices of size 4x512 if we have 1 head:
512 512 512
+-----------+ +-----------+ +-----------+
| Q' | | K' | | V' |
4 | | 4 | | 4 | |
| | | | | |
| | | | | |
+-----------+ +-----------+ +-----------+
But if we have two heads we will split these into multiple matrices:
256 256 256 256 256 256
+-----+ +-----+ +-----+ +-----+ +-----+ +-----+
| Q'_1| | Q'_2| | K'_1| | K'_2| | V'_1| | V'_2|
4 | | | | | | | | | | | |
| | | | | | | | | | | |
| | | | | | | | | | | |
+-----+ +-----+ +-----+ +-----+ +-----+ +-----+
Then we take each of these and perform the single head attention on each of them separatly:
head1 = Attention(Q'_1, K'_1, V'_1) = softmax((Q'_1, K'_1, V'_1)/√dₖ) x V'_1
head2 = Attention(Q'_2, K'_2, V'_2) = softmax((Q'_2, K'_2, V'_2)/√dₖ) x V'_2
The we contatenate those heads and multiple by W^o.
So for example, if we want to have 4 heads and the embedding dimension size is 512, then we will have 4 4x128 matrices. Each one of these are called a head and they are separate from each other, and are used to perform the single-head attention function that we went through above.
+----------------+ +-------+ +--------+
| Q | | Key | | Q' |
| | X | | = | |
| (4, 128) | |(128,4)| | (4, 4) |
+----------------+ | | +--------+
| |
| |
+-------+
Attention(Q'₀, K'₀, V'₀) = softmax((Q'₀, K'₀, V'₀)/√dₖ) x V'₀
Attention(Q'₁, K'₁, V'₁) = softmax((Q'₁, K'₁, V'₁)/√dₖ) x V'₁
Attention(Q'₂, K'₂, V'₂) = softmax((Q'₂, K'₂, V'₂)/√dₖ) x V'₂
Attention(Q'₃, K'₃, V'₃) = softmax((Q'₃, K'₃, V'₃)/√dₖ) x V'₃
Those will output 4 (sequence_length x dₖ
) matrices. So why would we want to do
this?
Well, notice how each attention calculation will still be using all the words/
tokens of the input sequence, 4 above, but uses fewer dimensions than with the
single head attention (128). This has implication for the softmax calculation
which now only sees a subset of the embedding dimension values. This is what
allows each of the heads to "focus" on different parts of the dimension space
and it is what causes the model to learn different things about the input
sequence.
These matrices are then concatenated into a single matrix:
+---------+
Concat(head_1, ..., head_h) = | H |
| (4, 512)|
+---------+
And this matrix is then multiplied by a learnable parameter matrix W^o:
+---------+ +-----------+ +-------+ (MH-A=MultiHead-Attention)
| H | | W^o | | MH-A |
| (4, 512)| X | (512, 512)| = |(4,512)|
+---------+ +-----------+ +-------+
Notice that we did not have additional matrices in the single head attention model.
In multi-head attention (MHA) we have multiple heads and each head has its own query, key, and value matrices:
+-----+ +-----+ +-----+ +-----+
| Q'_1| | Q'_2| | Q'_3| | Q'_4|
| | | | | | | |
| | | | | | | |
| | | | | | | |
+-----+ +-----+ +-----+ +-----+
+-----+ +-----+ +-----+ +-----+
| K'_1| | K'_2| | K'_3| | K'_4|
| | | | | | | |
| | | | | | | |
| | | | | | | |
+-----+ +-----+ +-----+ +-----+
+-----+ +-----+ +-----+ +-----+
| V'_1| | V'_2| | V'_3| | V'_4|
| | | | | | | |
| | | | | | | |
| | | | | | | |
+-----+ +-----+ +-----+ +-----+
We will still have a number of query heads (h) but only a single key and values vector:
+-----+ +-----+ +-----+ +-----+
| Q'_1| | Q'_2| | Q'_3| | Q'_4|
| | | | | | | |
| | | | | | | |
| | | | | | | |
+-----+ +-----+ +-----+ +-----+
+-----+
| K |
| |
| |
| |
+-----+
+-----+
| V |
| |
| |
| |
+-----+
There is also a grouped query attention (GQA) where we have the same number of query heads but we group them:
+-----+ +-----+ +-----+ +-----+
| Q'_1| | Q'_2| | Q'_3| | Q'_4|
| | | | | | | |
| | | | | | | |
| | | | | | | |
+-----+ +-----+ +-----+ +-----+
+-----+ +-----+
| K | | K |
| | | |
| | | |
| | | |
+-----+ +-----+
+-----+ +-----+
| V | | V |
| | | |
| | | |
| | | |
+-----+ +-----+
The Query vector is information that we are looking for, like the reason for saying something when we speak. So since we multiply X with W_Q to get Q, the Q vector has a context of sort, which in the case of Query is the information about the token and how it relates to the overall goal of the sentence.
The Key vectors is the relevance of the word(s), represented by X, to the query. So the key vector how relevant this token is to the query.
Again since we mulitply X with W_K to get K, K also has a context which is the relevance of X to the query.
So the Query and Key vectors encode context, that is how this token relates to the entire sequence of tokens inputted to the model (this if after the encoders have processed it).
Visualize Q vector space as vectors in two dimensions and we have three vectors, one for "I", one for "like", and one for "icecream". And we also have a vector space for K with 3 vectors. When we calculate Q x Kᵗ we are getting a new square matrix, and the values in this matrix contain the attention scores. What this is doing is calculating the distances between the key matrix vectors to the query vector (just one). This can be done by looking at the angle between the vectors or calculating the dot product.
Smaller values in the attention score mean that we should pay less attention to them and larger values mean that we should pay more attention to those tokens.
Next the output attention layer is passed to the Add&Norm layer which take it as input and also takes a copy of the Value matrix which is passed around the attention layer. This is what is called a skip connection or a residual connection.
We also have multi-head attention as described above in the decoder but there
is another layer called masked multi-head attention. This is while training,
well it is also used during inference but bare with me, where if we have a
translation task, the input to the decoder is the target sequence (the
translated version of the input sequence to the encoder). But we don't want the
decoders attention mechanism to take into account tokens that are ahead of the
current token.
Lets say we are training a model and the input sequence is "Dan älskar glass"
which is Swedish for "Dan loves icecream" which is the target sequence which is
the input to the decoder. We don't want the computation of the first token Dan
to take into account any of the tokens ahead of it, like "loves", "ice", and
"cream". So we mask those tokens out. This is done by setting the attention
scores for those tokens to negative infinity. This will cause the softmax
function to output 0 for those tokens:
+-------------------+
Dan |0.9| ~inf|~inf|~inf|
|-------------------|
loves |0.3| 0.9 |~inf|~inf|
|-------------------|
ice |0.1| 2.3 |0.9 |~inf|
|-------------------|
cream |2.0| -1.5|0.2 |0.9 |
+-------------------+
When performing inference the input to the decoder is the start of sequence token and nothing else, and the decoder will generate the next token in the sequence and add that to the input and continue like that. In this case there no future tokens to mask but this is done for consistency (I think).
Encoders are used for classification and regression tasks. Decoders are used for text generation tasks, like translation and summarization.
Encoder-Decoder are used for task like generative text like translation or summarization. So if we wanted to train a model to translate a sentence from English to Swedish, we would have the English sentence as input and the Swedish sentence as the output. The model would then be trained to predict the Swedish sentence given the English sentence.
Now, lets take a closer look at the boxes of the above diagram. I've written about embeddings in embeddings.md and positional encoding in positional-encoding.md so lets skip them for now and start with the encoder layer. So the next layer, or most often multiple layers, is the multi-head attention.
Standard attention uses 3 martixes, a query matrix, a key matrix, and a value matrix.
Let's start with the following input sentence "Dan loves icecream". The first step it split this into tokens, so we will have might get 4 tokens:
["Dan", "loves", "ice", "cream"]
Next, these words are mapped to token id from the model's vocabulary:
[1003] [82] [371] [10004]
Now, the model will the take these input and map them into embeddings which might be of a dimension of say 512. So we will have 4 vectors of 512 dimensions
'Dan' 1003 [0 ... 512]
'loves' 82 [0 ... 512]
'ice' 371 [0 ... 512]
'cream' 10004 [0 ... 512]
If the same word appears multiple times in the input, the same embedding will be used for each occurance. So there is currently no context or association between these words/token embeddings. They only contain information about each word/token itself, and nothing about the context in which it appears.
So with these embeddings the first thing in the model is to add a positional encoding to each of the embeddings. In the original paper this used absolute position encoding.
So we have our input matrix which in our case is a 4x512 matrix, where each entry is one of the tokens in the input sentence. We take this matrix and make four copies, but we will only focus on the first three for now:
+--------+ +-------+
| Input | -------> | Query |
| | +-------+
+--------+ -------> +-------+
| Key |
+-------+
+-------+
-------> | Value |
+-------+
+--------------+
--------------------------------> | Feed forward |
+--------------+
The attention function looks like this:
Attention(Q, K, V) = softmax((Q x Kᵗ)/√embedding_dim) x V
embedding_dim = 512 in our example
Lets start with looking at the nominator which is the matrix multiplication of Q and K transposed:
+----------------+ +-------+ +--------+
| Q | | Key | | |
| | X | | = | |
| (4, 512) | |(512,4)| | (4, 4) |
+----------------+ | | +--------+
| |
| |
+-------+
So, lets just think about this for a second. We know we copied the input matrix to Q and K.
This is a layer(s) that is used in transformers. It is used to normalize the output of the attention layer. It is called Add&Norm because it adds the output of the attention layer to the input (value matrix) and then normalizes it. What is normalization? I've written about this in normalization.md.
In the encoder, the first Add&Norm layer comes after the multi-head attention layer. There are two inputs into this layer, the first is the original value matrix which notice is passed around the multihead attention layer (in the diagram we saw earlier), this is called a residual connection or a skip connectionr. The second input is the output of the multihead attention layer. These are simply added together:
v_mha = V + MHA
V = Value matrix.
MHA = Multi-Head Attention output
This is residual connection/skip connection.
So v_mha
is a matrix where each row represents one transformed input token:
'Dan': [0.1, 0.2, 0.3, 0.4]
'loves': [0.5, 0.6, 0.7, 0.8]
'ice': [0.9, 1.0, 1.1, 1.2]
'cream': [1.3, 1.4, 1.5, 1.6]
D = 4 (is the dimension of the embeddings for each token)
Lets take one row, which is how layer normalization is performed:
Calculate the mean:
μ₀ = 1/D ∑ 0.1 + 0.2 + 0.3 + 0.4
μ₀ = 1/4 1 = 0.25
Calculate the variance:
σ²₀ = 1/D ∑ (0.1 - 0.25)² + (0.2 - 0.25)² + (0.3 - 0.25)² + (0.4 - 0.25)²
σ²₀ = 1/4 (-0.15)² + (-0.05)² + 0.05² + 0.15²)
σ²₀ = 1/4 (0.0225 + 0.0025 + 0.0025 + 0.0225)
σ²₀ = 1/4 (0.05)
σ²₀ = 0.0125
And then normalize:
0.1 - 0.25
---------- = -1.3411
√0.0125
So we do this for the entire first row which will produce the following values:
'Dan': [-1.3411, -0.4470, 0.4470, 1.3411]
By doing this for each token indepentently we make sure that the model looks at
the word Dan
and understands it in its own right, not influenced by other
tokens.
'Dan': [-1.3411, -0.4470, 0.4470, 1.3411]
'loves': [-1.3411, -0.4470, 0.4470, 1.3411]
'ice': [-1.3411, -0.4470, 0.4470, 1.3411]
'cream': [-1.3411, -0.4470, 0.4470, 1.3411]
These values are the same just because of the dummy values that I initialized used and would normally differ.
To understand this a little better I think it might help to look at what batch normalization would look like as well. In this case we don't take each row but instead each column, starting with the first one:
'Dan': [0.1, 0.2, 0.3, 0.4]
'loves': [0.5, 0.6, 0.7, 0.8]
'ice': [0.9, 1.0, 1.1, 1.2]
'cream': [1.3, 1.4, 1.5, 1.6]
So that the values produces for the first column will include values from all the tokens, 'Dan' 0.1, 'loves' 0.5, 'ice' 0.9, and 'cream' 1.3:
Calculate the mean:
μ₀ = 1/D ∑ 0.1 + 0.5 + 0.9 + 1.3
μ₀ = 1/4 2.8
μ₀ = 0.7
Calculate the variance:
σ²₀ = 1/D ∑ (0.1 - 0.7)² + (0.5 - 0.7)² + (0.9 - 0.7)² + (1.3 - 0.7)²
σ²₀ = 1/4 (-0.6)² + (-0.2)² + 0.2² + 0.6²)
σ²₀ = 1/4 (0.36 + 0.04 + 0.04 + 0.36)
σ²₀ = 1/4 (0.8)
σ²₀ = 0.2
And then normalize:
0.1 - 0.7
--------- = -1.3416
√0.2
0.5 - 0.7
--------- = -0.4472
√0.2
0.9 - 0.7
--------- = 0.4472
√0.2
1.3 - 0.7
--------- = 1.3416
√0.2
'Dan': [-1.3416]
'loves': [-0.4472]
'ice': [0.4472]
'cream': [1.3416]
So in batch normalization, the tokens can influence each other, but this is not the case in layer normalization as each token is handled completely separably.
In the transformer acrhitecture, we have the multi-head attention layer, followed by a Add&Norm layer, and then we have a feedforward layer.
This layer has two linear transformations with a ReLU activation function in between them. The first linear transformation expands the dimension of the input matrix, and the activation (like ReLU or GELU) function is applied to each element of the matrix. The second linear transformation reduces the dimension back to the original. I've seen names for the tensors in places, for example in llama.cpp where up (ffn_up) is used for the first transformation which expands the dimentions, gate (ffn_gate) is used as the activation function (I'm not sure why the name is gate but perhaps this is because the activation function will prevent values from passing through depending on their value), and down (ffn_down) is used for the second transformation which reduces the dimension back to the original.
So lets say that the output from the Add&Norm layer is:
'Dan': [0.1, 0.2, 0.3, 0.4]
'loves': [0.5, 0.6, 0.7, 0.8]
'ice': [0.9, 1.0, 1.1, 1.2]
'cream': [1.3, 1.4, 1.5, 1.6]
This layer transforms the dimesion of the vector from 4 to lets say 8. So we would have a weight matrix of size 4x8 which the above matrix would be multiplied with creating a 8x8 matrix.
(4x4) FeedForward Weight Matrix (4x8)
+--+--+--+--+ +--+--+--+--+--+--+--+--+
Dan | | | | | | | | | | | | | |
+--+--+--+--+ +--+--+--+--+--+--+--+--+
loves | | | | | | | | | | | | | |
+--+--+--+--+ +--+--+--+--+--+--+--+--+
ice | | | | | | | | | | | | | |
+--+--+--+--+ +--+--+--+--+--+--+--+--+
cream | | | | | | | | | | | | | |
+--+--+--+--+ +--+--+--+--+--+--+--+--+
|-----------|
↑
new dimensions
An activation function is then applied to each element of the 4x8 matrix. The increased dimension in the feedforward layer of a Transformer model is achieved through a weight matrix, and the values in this matrix are learned during the training process. These values become part of the model's learned parameters (weights).
This is done so that the model can learn more complex relationships between the tokens.
This 4x8 matrix is then passed to a second linear transformation which usually reduces the dimension back to 4. So this would be a 4x8 matrix multiplied by a 8x4 matrix which would produce a 4x4 matrix.
(4x8) FeedForward Weight Matrix 2 (8x4)
+--+--+--+--+--+--+--+--+ +--+--+--+--+
Dan | | | | | | | | | | | | | |
+--+--+--+--+--+--+--+--+ +--+--+--+--+
loves | | | | | | | | | | | | | |
+--+--+--+--+--+--+--+--+ +--+--+--+--+
ice | | | | | | | | | | | | | |
+--+--+--+--+--+--+--+--+ +--+--+--+--+
cream | | | | | | | | | | | | | |
+--+--+--+--+--+--+--+--+ +--+--+--+--+
| | | | |
+--+--+--+--+
| | | | |
+--+--+--+--+
| | | | |
+--+--+--+--+
| | | | |
+--+--+--+--+
And this will result in a 4x4 matrix, the same size as before the feedforward layer. This matrix has been transformed from the higher-dimensional space back to the original dimensionality, but with values that have been processed through both layers of the feedforward network.
In LLama 2 the context length is 4096 bytes (which is 32Kb). This is the dimension that will be expanded from. The size of the dimension that this gets expanded to is often called the hidden dimension.
llm_load_print_meta: vocab type = SPM
llm_load_print_meta: n_vocab = 32000
llm_load_print_meta: n_ctx_train = 2048
llm_load_print_meta: n_embd = 4096
llm_load_print_meta: n_head = 32
llm_load_print_meta: n_head_kv = 32
llm_load_print_meta: n_layer = 32
llm_load_print_meta: n_rot = 128
llm_load_print_meta: n_embd_head_k = 128
llm_load_print_meta: n_embd_head_v = 128
llm_load_print_meta: n_gqa = 1
llm_load_print_meta: n_embd_k_gqa = 4096
llm_load_print_meta: n_embd_v_gqa = 4096
llm_load_print_meta: n_ff = 11008
n_ff
specifes the size of the hidden dimension. So it will take expand from
[1, 4096] x [4096, 11008] = [1, 11008], perform the non-linear operation and
then reduce it back to [1, 4096]. Recall that the feedforward layer operates
on a row of the embeddings, which represents a token of the input sequence.
After all the vectors have been processed they will then be concatenated into
a matrix of the same size as the input matrix.
This is a token in the embedding which stands for classification. This is used to represent the entire input sequence. From ChatGPT4:
It's important to note that despite the "classification" origin of its name, the
"CLS" token is used in all tasks where we need an aggregate representation of
the entire sequence, whether it's classification, named entity recognition, etc.
For example, if we're classifying a sentence as positive or negative
(sentiment analysis), we would feed the sentence into BERT, which would add the
"CLS" token to the start of the sentence. BERT processes the entire sequence and
outputs a vector for each token, but we would only use the vector corresponding
to the "CLS" token to make the positive/negative prediction.
Bert-small has 4 encoders and 15M learnable parameters. Bert-base has 12 encoders and 110M learnable parameters. Bert-large has 24 encoders and 340M learnable parameters.
Text-to-Text Transfer Transformer (T5) is a transformer model that is trained.
This is part of transformers which enable them to tokenize words that are not
in their vocabulary. This is done by using a subword tokenizer where a word is
split into two parts where both are part of the vocabulary but the complete word
by itself.
For example: tokenization
would be spit into token
##ization
where ##
is a special token that indicates that the previous token is a subword of a
complete word.
The computational complexity is proportional to the square of the sequence length (N²), where N is the number of tokens in the sequence. This means that as the sequence gets longer, the amount of computation (and memory) required grows quadratically. To understand why it's N², consider that for each token in the sequence, the self-attention mechanism computes a score with every other token, including itself. So, if you have N tokens, each token attends to N tokens, leading to N x N comparisons or computations.
The self-attention mechanism can be decomposed into vector operations. First we have the original:
Att(Q, K, V) = softmax(QKᵗ/√dₖ) x V
Now, we can describe the above for each specific element t
in the sequence:
Σ exp(qₜᵗkᵢ) . vᵢ
Att(Q, K, V)ₜ = -----------------
Σ exp(qₜkᵢ)
Just to be clear on the order of operations:
The summation in the numerator: for each key kᵢ, the exponentiated score exp(qₜᵗkᵢ) is multiplied by the corresponding value vᵢ. This multiplication happens after calculating the exponentiated scores but before the final division (normalization). The division/normilization is the final step of the softmax.
qₜ = a row from Q:
+--+--+--+--+--+--+--+
| | | | | | | |
+--+--+--+--+--+--+--+
0 511
kᵢ = a column from K:
+--+--+--+--+--+--+--+
| | | | | | | |
+--+--+--+--+--+--+--+
0 511
qᵗ kᵢ
+--+ +--+--+--+--+--+ +--+
| | x | | | | | | = |dp|
| | +--+--+--+--+--+ +--+
| |
| |
| |
| |
+--+
exp(dp) * vᵢ
This is done for each vector kᵢ and then summed up and used as the nominator. The denominator is also calculated so that we can normalize the values. So is this like performing the softmax (the division part) like the original version but where we do it for one row of the sequence matrix at a time.
After output of each iteration is an output vector and then all these are concatenated together to form the output matrix.
This would allow us to perform the operations without having to have the entire Q,K, and V matrices in memory at the same time. Instead, here we are computing one row of the matrix at a time. And we avoid having to store the entire attention score matrix (from QKᵗ) in memory.
But while this method can be more memory-efficient and might allow us to process much larger sequences, it might not be as computationally efficient as processing the entire matrix at once (QKᵗ).
While I've always looked at Q as the query if we look at the following equation qₜ is just a row of values and does not really 'have' to represent the query. It could be anything:
Σ exp(qₜᵗkᵢ) . vᵢ
Att(Q, K, V)ₜ = -----------------
Σ exp(qₜkᵢ)
I mention this as it was not obvious to me and because there are other possible way to represent the attention as we will see below.
Is is also possible to represent a variation of this where we don't have a query matrix but instead replaced/transformed it by a weight's matrix. So it is not comparing a set of query vectors with a set of key vectors but instead using a set of predefined weights:
Σ exp(Wₜᵢ+kᵢ) . vᵢ
Att+(W, K, V) = -----------------
Σ exp(Wₜᵢ+kᵢ)
There are multiple variants of attention. There is the one in the original paper where query, key, and value matrix per layer. There is also one where we split the