ggml.go 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255
  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. )
  31. func fileType(fileType uint32) string {
  32. switch fileType {
  33. case fileTypeF32:
  34. return "F32"
  35. case fileTypeF16:
  36. return "F16"
  37. case fileTypeQ4_0:
  38. return "Q4_0"
  39. case fileTypeQ4_1:
  40. return "Q4_1"
  41. case fileTypeQ4_1_F16:
  42. return "Q4_1_F16"
  43. case fileTypeQ8_0:
  44. return "Q8_0"
  45. case fileTypeQ5_0:
  46. return "Q5_0"
  47. case fileTypeQ5_1:
  48. return "Q5_1"
  49. case fileTypeQ2_K:
  50. return "Q2_K"
  51. case fileTypeQ3_K_S:
  52. return "Q3_K_S"
  53. case fileTypeQ3_K_M:
  54. return "Q3_K_M"
  55. case fileTypeQ3_K_L:
  56. return "Q3_K_L"
  57. case fileTypeQ4_K_S:
  58. return "Q4_K_S"
  59. case fileTypeQ4_K_M:
  60. return "Q4_K_M"
  61. case fileTypeQ5_K_S:
  62. return "Q5_K_S"
  63. case fileTypeQ5_K_M:
  64. return "Q5_K_M"
  65. case fileTypeQ6_K:
  66. return "Q6_K"
  67. default:
  68. return "unknown"
  69. }
  70. }
  71. type model interface {
  72. ModelFamily() string
  73. ModelType() string
  74. FileType() string
  75. NumLayers() int64
  76. }
  77. type container interface {
  78. Name() string
  79. Decode(*readSeekOffset) (model, error)
  80. }
  81. type containerGGML struct{}
  82. func (c *containerGGML) Name() string {
  83. return "ggml"
  84. }
  85. func (c *containerGGML) Decode(ro *readSeekOffset) (model, error) {
  86. // file contents aren't decoded
  87. ro.Seek(0, io.SeekEnd)
  88. return nil, nil
  89. }
  90. type containerGGMF struct {
  91. version uint32
  92. }
  93. func (c *containerGGMF) Name() string {
  94. return "ggmf"
  95. }
  96. func (c *containerGGMF) Decode(ro *readSeekOffset) (model, error) {
  97. var version uint32
  98. binary.Read(ro, binary.LittleEndian, &version)
  99. switch version {
  100. case 1:
  101. default:
  102. return nil, errors.New("invalid version")
  103. }
  104. c.version = version
  105. // remaining file contents aren't decoded
  106. ro.Seek(0, io.SeekEnd)
  107. return nil, nil
  108. }
  109. type containerGGJT struct {
  110. version uint32
  111. }
  112. func (c *containerGGJT) Name() string {
  113. return "ggjt"
  114. }
  115. func (c *containerGGJT) Decode(ro *readSeekOffset) (model, error) {
  116. var version uint32
  117. binary.Read(ro, binary.LittleEndian, &version)
  118. switch version {
  119. case 1, 2, 3:
  120. default:
  121. return nil, errors.New("invalid version")
  122. }
  123. c.version = version
  124. // different model types may have different layouts for hyperparameters
  125. var llama llamaModel
  126. binary.Read(ro, binary.LittleEndian, &llama.hyperparameters)
  127. // remaining file contents aren't decoded
  128. ro.Seek(0, io.SeekEnd)
  129. return &llama, nil
  130. }
  131. type containerLORA struct {
  132. version uint32
  133. }
  134. func (c *containerLORA) Name() string {
  135. return "ggla"
  136. }
  137. func (c *containerLORA) Decode(ro *readSeekOffset) (model, error) {
  138. var version uint32
  139. binary.Read(ro, binary.LittleEndian, &version)
  140. switch version {
  141. case 1:
  142. default:
  143. return nil, errors.New("invalid version")
  144. }
  145. c.version = version
  146. // remaining file contents aren't decoded
  147. ro.Seek(0, io.SeekEnd)
  148. return nil, nil
  149. }
  150. const (
  151. // Magic constant for `ggml` files (unversioned).
  152. FILE_MAGIC_GGML = 0x67676d6c
  153. // Magic constant for `ggml` files (versioned, ggmf).
  154. FILE_MAGIC_GGMF = 0x67676d66
  155. // Magic constant for `ggml` files (versioned, ggjt).
  156. FILE_MAGIC_GGJT = 0x67676a74
  157. // Magic constant for `ggla` files (LoRA adapter).
  158. FILE_MAGIC_GGLA = 0x67676C61
  159. // Magic constant for `gguf` files (versioned, gguf)
  160. FILE_MAGIC_GGUF_LE = 0x46554747
  161. FILE_MAGIC_GGUF_BE = 0x47475546
  162. )
  163. func DecodeGGML(r io.ReadSeeker) (*GGML, error) {
  164. ro := readSeekOffset{ReadSeeker: r}
  165. var magic uint32
  166. if err := binary.Read(&ro, binary.LittleEndian, &magic); err != nil {
  167. return nil, err
  168. }
  169. var c container
  170. switch magic {
  171. case FILE_MAGIC_GGML:
  172. c = &containerGGML{}
  173. case FILE_MAGIC_GGMF:
  174. c = &containerGGMF{}
  175. case FILE_MAGIC_GGJT:
  176. c = &containerGGJT{}
  177. case FILE_MAGIC_GGLA:
  178. c = &containerLORA{}
  179. case FILE_MAGIC_GGUF_LE:
  180. c = &containerGGUF{bo: binary.LittleEndian}
  181. case FILE_MAGIC_GGUF_BE:
  182. c = &containerGGUF{bo: binary.BigEndian}
  183. default:
  184. return nil, errors.New("invalid file magic")
  185. }
  186. model, err := c.Decode(&ro)
  187. if err != nil {
  188. return nil, err
  189. }
  190. // final model type
  191. return &GGML{
  192. container: c,
  193. model: model,
  194. Size: ro.offset,
  195. }, nil
  196. }
  197. type readSeekOffset struct {
  198. io.ReadSeeker
  199. offset int64
  200. }
  201. func (rso *readSeekOffset) Seek(offset int64, whence int) (int64, error) {
  202. offset, err := rso.ReadSeeker.Seek(offset, whence)
  203. if err != nil {
  204. return 0, err
  205. }
  206. rso.offset = offset
  207. return offset, nil
  208. }
  209. func (rso *readSeekOffset) Read(p []byte) (int, error) {
  210. n, err := rso.ReadSeeker.Read(p)
  211. rso.offset += int64(n)
  212. return n, err
  213. }