ggml.go 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242
  1. package llm
  2. import (
  3. "encoding/binary"
  4. "errors"
  5. "io"
  6. )
  7. type GGML struct {
  8. container
  9. model
  10. Size int64
  11. }
  12. const (
  13. fileTypeF32 uint32 = iota
  14. fileTypeF16
  15. fileTypeQ4_0
  16. fileTypeQ4_1
  17. fileTypeQ4_1_F16
  18. fileTypeQ8_0 uint32 = iota + 2
  19. fileTypeQ5_0
  20. fileTypeQ5_1
  21. fileTypeQ2_K
  22. fileTypeQ3_K_S
  23. fileTypeQ3_K_M
  24. fileTypeQ3_K_L
  25. fileTypeQ4_K_S
  26. fileTypeQ4_K_M
  27. fileTypeQ5_K_S
  28. fileTypeQ5_K_M
  29. fileTypeQ6_K
  30. fileTypeIQ2_XXS
  31. fileTypeIQ2_XS
  32. fileTypeQ2_K_S
  33. fileTypeQ3_K_XS
  34. fileTypeIQ3_XXS
  35. )
  36. func fileType(fileType uint32) string {
  37. switch fileType {
  38. case fileTypeF32:
  39. return "F32"
  40. case fileTypeF16:
  41. return "F16"
  42. case fileTypeQ4_0:
  43. return "Q4_0"
  44. case fileTypeQ4_1:
  45. return "Q4_1"
  46. case fileTypeQ4_1_F16:
  47. return "Q4_1_F16"
  48. case fileTypeQ8_0:
  49. return "Q8_0"
  50. case fileTypeQ5_0:
  51. return "Q5_0"
  52. case fileTypeQ5_1:
  53. return "Q5_1"
  54. case fileTypeQ2_K:
  55. return "Q2_K"
  56. case fileTypeQ3_K_S:
  57. return "Q3_K_S"
  58. case fileTypeQ3_K_M:
  59. return "Q3_K_M"
  60. case fileTypeQ3_K_L:
  61. return "Q3_K_L"
  62. case fileTypeQ4_K_S:
  63. return "Q4_K_S"
  64. case fileTypeQ4_K_M:
  65. return "Q4_K_M"
  66. case fileTypeQ5_K_S:
  67. return "Q5_K_S"
  68. case fileTypeQ5_K_M:
  69. return "Q5_K_M"
  70. case fileTypeQ6_K:
  71. return "Q6_K"
  72. case fileTypeIQ2_XXS:
  73. return "IQ2_XXS"
  74. case fileTypeIQ2_XS:
  75. return "IQ2_XS"
  76. case fileTypeQ2_K_S:
  77. return "Q2_K_S"
  78. case fileTypeQ3_K_XS:
  79. return "Q3_K_XS"
  80. case fileTypeIQ3_XXS:
  81. return "IQ3_XXS"
  82. default:
  83. return "unknown"
  84. }
  85. }
  86. type model interface {
  87. ModelFamily() string
  88. ModelType() string
  89. FileType() string
  90. NumLayers() uint32
  91. NumGQA() uint32
  92. NumEmbed() uint32
  93. NumHead() uint32
  94. NumHeadKv() uint32
  95. NumCtx() uint32
  96. }
  97. type KV map[string]any
  98. type Tensor struct {
  99. Name string
  100. Kind uint32
  101. Offset uint64
  102. // Shape is the number of elements in each dimension
  103. Shape []uint64
  104. io.WriterTo
  105. }
  106. func (t Tensor) blockSize() uint64 {
  107. switch {
  108. case t.Kind < 2:
  109. return 1
  110. case t.Kind < 10:
  111. return 32
  112. default:
  113. return 256
  114. }
  115. }
  116. func (t Tensor) typeSize() uint64 {
  117. blockSize := t.blockSize()
  118. switch t.Kind {
  119. case 0: // FP32
  120. return 4
  121. case 1: // FP16
  122. return 2
  123. case 2: // Q4_0
  124. return 2 + blockSize/2
  125. case 3: // Q4_1
  126. return 2 + 2 + blockSize/2
  127. case 6: // Q5_0
  128. return 2 + 4 + blockSize/2
  129. case 7: // Q5_1
  130. return 2 + 2 + 4 + blockSize/2
  131. case 8: // Q8_0
  132. return 2 + blockSize
  133. case 9: // Q8_1
  134. return 4 + 4 + blockSize
  135. case 10: // Q2_K
  136. return blockSize/16 + blockSize/4 + 2 + 2
  137. case 11: // Q3_K
  138. return blockSize/8 + blockSize/4 + 12 + 2
  139. case 12: // Q4_K
  140. return 2 + 2 + 12 + blockSize/2
  141. case 13: // Q5_K
  142. return 2 + 2 + 12 + blockSize/8 + blockSize/2
  143. case 14: // Q6_K
  144. return blockSize/2 + blockSize/4 + blockSize/16 + 2
  145. case 15: // Q8_K
  146. return 2 + blockSize + 2*blockSize/16
  147. case 16: // IQ2_XXS
  148. return 2 + 2*blockSize/8
  149. case 17: // IQ2_XS
  150. return 2 + 2*blockSize/8 + blockSize/32
  151. case 18: // IQ3_XXS
  152. return 2 + 3*blockSize/8
  153. default:
  154. return 0
  155. }
  156. }
  157. func (t Tensor) parameters() uint64 {
  158. var count uint64 = 1
  159. for _, n := range t.Shape {
  160. count *= n
  161. }
  162. return count
  163. }
  164. func (t Tensor) size() uint64 {
  165. return t.parameters() * t.typeSize() / t.blockSize()
  166. }
  167. type container interface {
  168. Name() string
  169. Decode(io.ReadSeeker) (model, error)
  170. }
  171. const (
  172. // Magic constant for `ggml` files (unversioned).
  173. FILE_MAGIC_GGML = 0x67676d6c
  174. // Magic constant for `ggml` files (versioned, ggmf).
  175. FILE_MAGIC_GGMF = 0x67676d66
  176. // Magic constant for `ggml` files (versioned, ggjt).
  177. FILE_MAGIC_GGJT = 0x67676a74
  178. // Magic constant for `ggla` files (LoRA adapter).
  179. FILE_MAGIC_GGLA = 0x67676C61
  180. // Magic constant for `gguf` files (versioned, gguf)
  181. FILE_MAGIC_GGUF_LE = 0x46554747
  182. FILE_MAGIC_GGUF_BE = 0x47475546
  183. )
  184. var ErrUnsupportedFormat = errors.New("unsupported model format")
  185. func DecodeGGML(rs io.ReadSeeker) (*GGML, error) {
  186. var magic uint32
  187. if err := binary.Read(rs, binary.LittleEndian, &magic); err != nil {
  188. return nil, err
  189. }
  190. var c container
  191. switch magic {
  192. case FILE_MAGIC_GGML, FILE_MAGIC_GGMF, FILE_MAGIC_GGJT:
  193. return nil, ErrUnsupportedFormat
  194. case FILE_MAGIC_GGLA:
  195. c = &containerGGLA{}
  196. case FILE_MAGIC_GGUF_LE:
  197. c = &containerGGUF{ByteOrder: binary.LittleEndian}
  198. case FILE_MAGIC_GGUF_BE:
  199. c = &containerGGUF{ByteOrder: binary.BigEndian}
  200. default:
  201. return nil, errors.New("invalid file magic")
  202. }
  203. model, err := c.Decode(rs)
  204. if errors.Is(err, io.EOF) {
  205. // noop
  206. } else if err != nil {
  207. return nil, err
  208. }
  209. offset, err := rs.Seek(0, io.SeekCurrent)
  210. if err != nil {
  211. return nil, err
  212. }
  213. // final model type
  214. return &GGML{
  215. container: c,
  216. model: model,
  217. Size: offset,
  218. }, nil
  219. }