model.go 6.18 KB
Newer Older
Michael Yang's avatar
llama4  
Michael Yang committed
1
2
3
4
5
package llama4

import (
	"bytes"
	"image"
Michael Yang's avatar
Michael Yang committed
6
	"slices"
Michael Yang's avatar
llama4  
Michael Yang committed
7
8
9
10
11
12
13
14
15
16
17
18

	"github.com/ollama/ollama/fs"
	"github.com/ollama/ollama/kvcache"
	"github.com/ollama/ollama/ml"
	"github.com/ollama/ollama/ml/nn"
	"github.com/ollama/ollama/model"
	"github.com/ollama/ollama/model/input"
)

type Model struct {
	model.Base
	model.BytePairEncoding
Michael Yang's avatar
Michael Yang committed
19
	ImageProcessor
Michael Yang's avatar
llama4  
Michael Yang committed
20

21
	*VisionModel `gguf:"v"`
Michael Yang's avatar
llama4  
Michael Yang committed
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
	*Projector   `gguf:"mm"`
	*TextModel
}

type Projector struct {
	Linear1 *nn.Linear `gguf:"linear_1"`
}

func (p *Projector) Forward(ctx ml.Context, visionOutputs ml.Tensor) ml.Tensor {
	return p.Linear1.Forward(ctx, visionOutputs)
}

func New(c fs.Config) (model.Model, error) {
	m := Model{
		BytePairEncoding: model.NewBytePairEncoding(
Michael Yang's avatar
Michael Yang committed
37
38
			c.String("tokenizer.ggml.pretokenizer",
				`[^\r\n\p{L}\p{N}]?[\p{Lu}\p{Lt}\p{Lm}\p{Lo}\p{M}]*[\p{Ll}\p{Lm}\p{Lo}\p{M}]+(?i:'s|'t|'re|'ve|'m|'ll|'d)?|[^\r\n\p{L}\p{N}]?[\p{Lu}\p{Lt}\p{Lm}\p{Lo}\p{M}]+[\p{Ll}\p{Lm}\p{Lo}\p{M}]*(?i:'s|'t|'re|'ve|'m|'ll|'d)?|\p{N}{1,3}| ?[^\s\p{L}\p{N}]+[\r\n/]*|\s*[\r\n]+|\s+(?!\S)|\s+`),
Michael Yang's avatar
llama4  
Michael Yang committed
39
40
			&model.Vocabulary{
				Values: c.Strings("tokenizer.ggml.tokens"),
Michael Yang's avatar
Michael Yang committed
41
				Types:  c.Ints("tokenizer.ggml.token_type"),
Michael Yang's avatar
llama4  
Michael Yang committed
42
43
				Merges: c.Strings("tokenizer.ggml.merges"),
				AddBOS: c.Bool("tokenizer.ggml.add_bos_token", true),
44
				BOS:    []int32{int32(c.Uint("tokenizer.ggml.bos_token_id"))},
Michael Yang's avatar
llama4  
Michael Yang committed
45
				AddEOS: c.Bool("tokenizer.ggml.add_eos_token", false),
46
47
48
49
				EOS: append(
					[]int32{int32(c.Uint("tokenizer.ggml.eos_token_id"))},
					c.Ints("tokenizer.ggml.eos_token_ids")...,
				),
Michael Yang's avatar
llama4  
Michael Yang committed
50
51
			},
		),
Michael Yang's avatar
Michael Yang committed
52
53
54
		ImageProcessor: newImageProcessor(c),
		VisionModel:    newVisionModel(c),
		TextModel:      newTextModel(c),
Michael Yang's avatar
llama4  
Michael Yang committed
55
56
57
	}

	m.Cache = kvcache.NewWrapperCache(
Michael Yang's avatar
Michael Yang committed
58
		kvcache.NewChunkedAttentionCache(int32(c.Uint("attention.chunk_size", 8192)), m.Shift),
Michael Yang's avatar
llama4  
Michael Yang committed
59
60
61
62
63
64
		kvcache.NewCausalCache(m.Shift),
	)

	return &m, nil
}

65
func (m *Model) EncodeMultimodal(ctx ml.Context, multimodalData []byte) ([]input.Multimodal, error) {
Michael Yang's avatar
llama4  
Michael Yang committed
66
67
68
69
70
71
72
73
74
	if len(m.VisionModel.Layers) < 1 {
		return nil, model.ErrNoVisionModel
	}

	img, _, err := image.Decode(bytes.NewReader(multimodalData))
	if err != nil {
		return nil, err
	}

Michael Yang's avatar
Michael Yang committed
75
	pixelsLocal, pixelsGlobal, size, err := m.ProcessImage(img)
Michael Yang's avatar
llama4  
Michael Yang committed
76
77
78
79
	if err != nil {
		return nil, err
	}

80
	tilesLocal := ctx.Input().FromFloatSlice(pixelsLocal, size.X, size.Y, m.numChannels)
Michael Yang's avatar
llama4  
Michael Yang committed
81

Michael Yang's avatar
Michael Yang committed
82
	ratioW, ratioH := size.X/m.imageSize, size.Y/m.imageSize
Michael Yang's avatar
Michael Yang committed
83
84
85
86
87
88
89
90

	tilesLocal = tilesLocal.Reshape(ctx, size.X/ratioW, ratioW, size.Y, m.numChannels).Permute(ctx, 0, 2, 1, 3).Contiguous(ctx)
	tilesLocal = tilesLocal.Reshape(ctx, size.X/ratioW*size.Y/ratioH, ratioH, ratioW, m.numChannels).Permute(ctx, 0, 3, 2, 1).Contiguous(ctx)
	tilesLocal = tilesLocal.Reshape(ctx, size.X/ratioW, size.Y/ratioH, m.numChannels, ratioH*ratioW)

	pixelValues := tilesLocal

	if len(pixelsGlobal) > 0 {
91
		tilesGlobal := ctx.Input().FromFloatSlice(pixelsGlobal, m.imageSize, m.imageSize, m.numChannels)
Michael Yang's avatar
Michael Yang committed
92
93
94
		pixelValues = pixelValues.Concat(ctx, tilesGlobal, 3)
	}

Michael Yang's avatar
llama4  
Michael Yang committed
95
96
	visionOutputs := m.VisionModel.Forward(ctx, pixelValues)
	visionOutputs = visionOutputs.Reshape(ctx, visionOutputs.Dim(0), visionOutputs.Dim(1)*visionOutputs.Dim(2)*visionOutputs.Dim(3))
Michael Yang's avatar
Michael Yang committed
97
98
	projectedOutputs := m.Projector.Forward(ctx, visionOutputs)

99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
	var multimodal []input.Multimodal
	aspectRatio := image.Point{ratioW, ratioH}

	var offset int
	patchesPerChunk := projectedOutputs.Dim(1)
	if aspectRatio.Y*aspectRatio.X > 1 {
		patchesPerChunk = projectedOutputs.Dim(1) / (aspectRatio.X*aspectRatio.Y + 1)

		for range aspectRatio.Y {
			for x := range aspectRatio.X {
				view := projectedOutputs.View(ctx, projectedOutputs.Stride(1)*offset,
					projectedOutputs.Dim(0), projectedOutputs.Stride(1),
					patchesPerChunk)
				var separator separator
				if x < aspectRatio.X-1 {
					separator.x = true // <|tile_x_separator|>
				} else {
					separator.y = true // <|tile_y_separator|>
				}
				multimodal = append(multimodal, input.Multimodal{Tensor: view, Data: &separator})
				offset += patchesPerChunk
			}
		}
	}
Michael Yang's avatar
Michael Yang committed
123

124
125
126
127
	view := projectedOutputs.View(ctx, projectedOutputs.Stride(1)*offset,
		projectedOutputs.Dim(0), projectedOutputs.Stride(1),
		patchesPerChunk)
	multimodal = append(multimodal, input.Multimodal{Tensor: view, Data: &separator{}})
Michael Yang's avatar
Michael Yang committed
128

129
	return multimodal, nil
Michael Yang's avatar
Michael Yang committed
130
131
}

132
133
134
type separator struct {
	x bool
	y bool
Michael Yang's avatar
llama4  
Michael Yang committed
135
136
}

137
138
func (m *Model) PostTokenize(inputs []*input.Input) ([]*input.Input, error) {
	var result []*input.Input
Michael Yang's avatar
Michael Yang committed
139
	for _, inp := range inputs {
140
		if len(inp.Multimodal) == 0 {
Michael Yang's avatar
Michael Yang committed
141
142
143
144
			result = append(result, inp)
			continue
		}

145
146
		var imageInputs []*input.Input
		imageInputs = append(imageInputs, &input.Input{Token: 200080}) // <|image_start|>
Michael Yang's avatar
Michael Yang committed
147

148
149
150
151
152
153
		for i, mm := range inp.Multimodal {
			patchesPerChunk := mm.Tensor.Dim(1)

			if i < len(inp.Multimodal)-1 {
				separator := mm.Data.(*separator)

154
155
				imageInputs = append(imageInputs, &input.Input{Token: 200092, Multimodal: []input.Multimodal{{Tensor: mm.Tensor}}, MultimodalHash: inp.MultimodalHash, SameBatch: patchesPerChunk}) // <|patch|>
				imageInputs = append(imageInputs, slices.Repeat([]*input.Input{{Token: 200092}}, patchesPerChunk-1)...)
Michael Yang's avatar
Michael Yang committed
156

157
				if separator.x {
158
					imageInputs = append(imageInputs, &input.Input{Token: 200084}) // <|tile_x_separator|>
159
160
				}
				if separator.y {
161
					imageInputs = append(imageInputs, &input.Input{Token: 200085}) // <|tile_y_separator|>
162
163
				}
			} else {
164
165
166
167
				imageInputs = append(imageInputs, &input.Input{Token: 200090})                                                                                                                      // <|image|>
				imageInputs = append(imageInputs, &input.Input{Token: 200092, Multimodal: []input.Multimodal{{Tensor: mm.Tensor}}, MultimodalHash: inp.MultimodalHash, SameBatch: patchesPerChunk}) // <|patch|>
				imageInputs = append(imageInputs, slices.Repeat([]*input.Input{{Token: 200092}}, patchesPerChunk-1)...)
				imageInputs = append(imageInputs, &input.Input{Token: 200080}) // <|image_end|>
Michael Yang's avatar
Michael Yang committed
168
169
170
171
172
173
174
			}
		}

		result = append(result, imageInputs...)
	}

	return result, nil
Michael Yang's avatar
Michael Yang committed
175
176
}

Michael Yang's avatar
llama4  
Michael Yang committed
177
func (m *Model) Forward(ctx ml.Context, batch input.Batch) (ml.Tensor, error) {
178
179
	positions := ctx.Input().FromIntSlice(batch.Positions, len(batch.Positions))
	outputs := ctx.Input().FromIntSlice(batch.Outputs, len(batch.Outputs))
Michael Yang's avatar
llama4  
Michael Yang committed
180
181
182
183
184
185
186

	return m.TextModel.Forward(ctx, batch.Inputs, positions, outputs, batch, m.Cache), nil
}

func init() {
	model.Register("llama4", New)
}