model.go 3.46 KB
Newer Older
Michael Yang's avatar
Michael Yang committed
1
2
3
package mllama

import (
4
5
	"bytes"
	"image"
6

7
	"github.com/ollama/ollama/fs"
Jesse Gross's avatar
Jesse Gross committed
8
	"github.com/ollama/ollama/kvcache"
Michael Yang's avatar
Michael Yang committed
9
10
11
	"github.com/ollama/ollama/ml"
	"github.com/ollama/ollama/ml/nn"
	"github.com/ollama/ollama/model"
12
	"github.com/ollama/ollama/model/input"
Michael Yang's avatar
Michael Yang committed
13
14
15
16
17
18
19
20
21
22
23
24
25
26
)

type Model struct {
	model.Base
	model.BytePairEncoding

	*VisionModel `gguf:"v,vision"`
	*TextModel

	Projector *nn.Linear `gguf:"mm.0"`

	ImageProcessor
}

Jesse Gross's avatar
Jesse Gross committed
27
28
29
30
31
const (
	crossAttentionLayer = iota
	selfAttentionLayer
)

32
func New(c fs.Config) (model.Model, error) {
Jesse Gross's avatar
Jesse Gross committed
33
	m := Model{
Michael Yang's avatar
Michael Yang committed
34
35
36
37
		BytePairEncoding: model.NewBytePairEncoding(
			c.String("tokenizer.ggml.pretokenizer", `(?i:'s|'t|'re|'ve|'m|'ll|'d)|[^\r\n\p{L}\p{N}]?\p{L}+|\p{N}{1,3}| ?[^\s\p{L}\p{N}]+[\r\n]*|\s*[\r\n]+|\s+(?!\S)|\s+`),
			&model.Vocabulary{
				Values: c.Strings("tokenizer.ggml.tokens"),
Michael Yang's avatar
Michael Yang committed
38
				Types:  c.Ints("tokenizer.ggml.token_type"),
Michael Yang's avatar
Michael Yang committed
39
				Merges: c.Strings("tokenizer.ggml.merges"),
40
				BOS:    int32(c.Uint("tokenizer.ggml.bos_token_id")),
41
				AddBOS: c.Bool("tokenizer.ggml.add_bos_token", true),
42
				EOS:    int32(c.Uint("tokenizer.ggml.eos_token_id")),
43
				AddEOS: c.Bool("tokenizer.ggml.add_eos_token", false),
Michael Yang's avatar
Michael Yang committed
44
45
46
				// TODO: set EOT to EOS otherwise 0 will stop generation
				EOT:    int32(c.Uint("tokenizer.ggml.eos_token_id")),
				AddEOT: c.Bool("tokenizer.ggml.add_eos_token", false),
Michael Yang's avatar
Michael Yang committed
47
48
49
50
51
			},
		),
		ImageProcessor: newImageProcessor(c),
		VisionModel:    newVisionModel(c),
		TextModel:      newTextModel(c),
Jesse Gross's avatar
Jesse Gross committed
52
53
	}

54
55
56
	encoderCache := kvcache.NewEncoderCache()
	encoderCache.SetConfig(ml.CacheConfig{})
	m.Cache = kvcache.NewWrapperCache(encoderCache, kvcache.NewCausalCache(m.TextModel.Shift))
Jesse Gross's avatar
Jesse Gross committed
57
58

	return &m, nil
Michael Yang's avatar
Michael Yang committed
59
60
}

61
func (m *Model) EncodeMultimodal(ctx ml.Context, multimodalData []byte) (any, error) {
62
63
64
65
	if len(m.VisionModel.Transformer.Layers) == 0 || len(m.GlobalTransformer.Layers) == 0 {
		return nil, model.ErrNoVisionModel
	}

66
67
68
69
	image, _, err := image.Decode(bytes.NewReader(multimodalData))
	if err != nil {
		return nil, err
	}
Michael Yang's avatar
Michael Yang committed
70

71
	f32s, ratio, err := m.ImageProcessor.ProcessImage(image)
72
73
74
	if err != nil {
		return nil, err
	}
Michael Yang's avatar
Michael Yang committed
75

76
	pixelValues, err := ctx.Input().FromFloatSlice(f32s, m.imageSize, m.imageSize, m.numChannels, ratio.numTiles())
77
78
79
	if err != nil {
		return nil, err
	}
Michael Yang's avatar
Michael Yang committed
80

81
82
83
	pixelValues = pixelValues.Pad(ctx, 0, 0, 0, m.ImageProcessor.maxNumTiles-ratio.numTiles())

	aspectRatio, err := ctx.Input().FromIntSlice([]int32{int32(ratio.rank)}, 1)
84
85
86
87
	if err != nil {
		return nil, err
	}

Michael Yang's avatar
arange  
Michael Yang committed
88
	positionIDs := ctx.Arange(0, 1601, 1, ml.DTypeI32)
89
90
91
92
	crossAttentionStates := m.VisionModel.Forward(ctx, pixelValues, positionIDs, aspectRatio)
	return m.Projector.Forward(ctx, crossAttentionStates), nil
}

93
func (m *Model) PostTokenize(inputs []input.Input) ([]input.Input, error) {
94
	for i := range inputs {
95
96
		if inputs[i].Multimodal != nil {
			inputs[i].Token = 128256 // <|image|>
Michael Yang's avatar
Michael Yang committed
97
		}
98
99
100
101
102
	}

	return inputs, nil
}

Jesse Gross's avatar
Jesse Gross committed
103
func (m *Model) Forward(ctx ml.Context, batch input.Batch) (ml.Tensor, error) {
104
	var crossAttentionStates ml.Tensor
Jesse Gross's avatar
Jesse Gross committed
105
	if len(batch.Multimodal) > 0 {
106
		crossAttentionStates = batch.Multimodal[len(batch.Multimodal)-1].Multimodal.(ml.Tensor)
Michael Yang's avatar
Michael Yang committed
107
108
	}

Jesse Gross's avatar
Jesse Gross committed
109
	positions, err := ctx.Input().FromIntSlice(batch.Positions, len(batch.Positions))
Michael Yang's avatar
Michael Yang committed
110
111
112
113
	if err != nil {
		return nil, err
	}

Jesse Gross's avatar
Jesse Gross committed
114
	outputs, err := ctx.Input().FromIntSlice(batch.Outputs, len(batch.Outputs))
Michael Yang's avatar
Michael Yang committed
115
116
117
118
	if err != nil {
		return nil, err
	}

119
	// TODO: attention mask, cross attention mask
120
	return m.TextModel.Forward(ctx, batch.Inputs, positions, outputs, crossAttentionStates, nil, m.Cache.(*kvcache.WrapperCache)), nil
Michael Yang's avatar
Michael Yang committed
121
122
123
124
125
}

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