size.go 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687
  1. // Protocol Buffers for Go with Gadgets
  2. //
  3. // Copyright (c) 2013, The GoGo Authors. All rights reserved.
  4. // http://github.com/gogo/protobuf
  5. //
  6. // Redistribution and use in source and binary forms, with or without
  7. // modification, are permitted provided that the following conditions are
  8. // met:
  9. //
  10. // * Redistributions of source code must retain the above copyright
  11. // notice, this list of conditions and the following disclaimer.
  12. // * Redistributions in binary form must reproduce the above
  13. // copyright notice, this list of conditions and the following disclaimer
  14. // in the documentation and/or other materials provided with the
  15. // distribution.
  16. //
  17. // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
  18. // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
  19. // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
  20. // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
  21. // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
  22. // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
  23. // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
  24. // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
  25. // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
  26. // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
  27. // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  28. /*
  29. The size plugin generates a Size or ProtoSize method for each message.
  30. This is useful with the MarshalTo method generated by the marshalto plugin and the
  31. gogoproto.marshaler and gogoproto.marshaler_all extensions.
  32. It is enabled by the following extensions:
  33. - sizer
  34. - sizer_all
  35. - protosizer
  36. - protosizer_all
  37. The size plugin also generates a test given it is enabled using one of the following extensions:
  38. - testgen
  39. - testgen_all
  40. And a benchmark given it is enabled using one of the following extensions:
  41. - benchgen
  42. - benchgen_all
  43. Let us look at:
  44. github.com/gogo/protobuf/test/example/example.proto
  45. Btw all the output can be seen at:
  46. github.com/gogo/protobuf/test/example/*
  47. The following message:
  48. option (gogoproto.sizer_all) = true;
  49. message B {
  50. option (gogoproto.description) = true;
  51. optional A A = 1 [(gogoproto.nullable) = false, (gogoproto.embed) = true];
  52. repeated bytes G = 2 [(gogoproto.customtype) = "github.com/gogo/protobuf/test/custom.Uint128", (gogoproto.nullable) = false];
  53. }
  54. given to the size plugin, will generate the following code:
  55. func (m *B) Size() (n int) {
  56. if m == nil {
  57. return 0
  58. }
  59. var l int
  60. _ = l
  61. l = m.A.Size()
  62. n += 1 + l + sovExample(uint64(l))
  63. if len(m.G) > 0 {
  64. for _, e := range m.G {
  65. l = e.Size()
  66. n += 1 + l + sovExample(uint64(l))
  67. }
  68. }
  69. if m.XXX_unrecognized != nil {
  70. n += len(m.XXX_unrecognized)
  71. }
  72. return n
  73. }
  74. and the following test code:
  75. func TestBSize(t *testing5.T) {
  76. popr := math_rand5.New(math_rand5.NewSource(time5.Now().UnixNano()))
  77. p := NewPopulatedB(popr, true)
  78. dAtA, err := github_com_gogo_protobuf_proto2.Marshal(p)
  79. if err != nil {
  80. panic(err)
  81. }
  82. size := p.Size()
  83. if len(dAtA) != size {
  84. t.Fatalf("size %v != marshalled size %v", size, len(dAtA))
  85. }
  86. }
  87. func BenchmarkBSize(b *testing5.B) {
  88. popr := math_rand5.New(math_rand5.NewSource(616))
  89. total := 0
  90. pops := make([]*B, 1000)
  91. for i := 0; i < 1000; i++ {
  92. pops[i] = NewPopulatedB(popr, false)
  93. }
  94. b.ResetTimer()
  95. for i := 0; i < b.N; i++ {
  96. total += pops[i%1000].Size()
  97. }
  98. b.SetBytes(int64(total / b.N))
  99. }
  100. The sovExample function is a size of varint function for the example.pb.go file.
  101. */
  102. package size
  103. import (
  104. "fmt"
  105. "os"
  106. "strconv"
  107. "strings"
  108. "github.com/gogo/protobuf/gogoproto"
  109. "github.com/gogo/protobuf/proto"
  110. descriptor "github.com/gogo/protobuf/protoc-gen-gogo/descriptor"
  111. "github.com/gogo/protobuf/protoc-gen-gogo/generator"
  112. "github.com/gogo/protobuf/vanity"
  113. )
  114. type size struct {
  115. *generator.Generator
  116. generator.PluginImports
  117. atleastOne bool
  118. localName string
  119. typesPkg generator.Single
  120. }
  121. func NewSize() *size {
  122. return &size{}
  123. }
  124. func (p *size) Name() string {
  125. return "size"
  126. }
  127. func (p *size) Init(g *generator.Generator) {
  128. p.Generator = g
  129. }
  130. func wireToType(wire string) int {
  131. switch wire {
  132. case "fixed64":
  133. return proto.WireFixed64
  134. case "fixed32":
  135. return proto.WireFixed32
  136. case "varint":
  137. return proto.WireVarint
  138. case "bytes":
  139. return proto.WireBytes
  140. case "group":
  141. return proto.WireBytes
  142. case "zigzag32":
  143. return proto.WireVarint
  144. case "zigzag64":
  145. return proto.WireVarint
  146. }
  147. panic("unreachable")
  148. }
  149. func keySize(fieldNumber int32, wireType int) int {
  150. x := uint32(fieldNumber)<<3 | uint32(wireType)
  151. size := 0
  152. for size = 0; x > 127; size++ {
  153. x >>= 7
  154. }
  155. size++
  156. return size
  157. }
  158. func (p *size) sizeVarint() {
  159. p.P(`
  160. func sov`, p.localName, `(x uint64) (n int) {
  161. for {
  162. n++
  163. x >>= 7
  164. if x == 0 {
  165. break
  166. }
  167. }
  168. return n
  169. }`)
  170. }
  171. func (p *size) sizeZigZag() {
  172. p.P(`func soz`, p.localName, `(x uint64) (n int) {
  173. return sov`, p.localName, `(uint64((x << 1) ^ uint64((int64(x) >> 63))))
  174. }`)
  175. }
  176. func (p *size) std(field *descriptor.FieldDescriptorProto, name string) (string, bool) {
  177. if gogoproto.IsStdTime(field) {
  178. if gogoproto.IsNullable(field) {
  179. return p.typesPkg.Use() + `.SizeOfStdTime(*` + name + `)`, true
  180. } else {
  181. return p.typesPkg.Use() + `.SizeOfStdTime(` + name + `)`, true
  182. }
  183. } else if gogoproto.IsStdDuration(field) {
  184. if gogoproto.IsNullable(field) {
  185. return p.typesPkg.Use() + `.SizeOfStdDuration(*` + name + `)`, true
  186. } else {
  187. return p.typesPkg.Use() + `.SizeOfStdDuration(` + name + `)`, true
  188. }
  189. }
  190. return "", false
  191. }
  192. func (p *size) generateField(proto3 bool, file *generator.FileDescriptor, message *generator.Descriptor, field *descriptor.FieldDescriptorProto, sizeName string) {
  193. fieldname := p.GetOneOfFieldName(message, field)
  194. nullable := gogoproto.IsNullable(field)
  195. repeated := field.IsRepeated()
  196. doNilCheck := gogoproto.NeedsNilCheck(proto3, field)
  197. if repeated {
  198. p.P(`if len(m.`, fieldname, `) > 0 {`)
  199. p.In()
  200. } else if doNilCheck {
  201. p.P(`if m.`, fieldname, ` != nil {`)
  202. p.In()
  203. }
  204. packed := field.IsPacked() || (proto3 && field.IsPacked3())
  205. _, wire := p.GoType(message, field)
  206. wireType := wireToType(wire)
  207. fieldNumber := field.GetNumber()
  208. if packed {
  209. wireType = proto.WireBytes
  210. }
  211. key := keySize(fieldNumber, wireType)
  212. switch *field.Type {
  213. case descriptor.FieldDescriptorProto_TYPE_DOUBLE,
  214. descriptor.FieldDescriptorProto_TYPE_FIXED64,
  215. descriptor.FieldDescriptorProto_TYPE_SFIXED64:
  216. if packed {
  217. p.P(`n+=`, strconv.Itoa(key), `+sov`, p.localName, `(uint64(len(m.`, fieldname, `)*8))`, `+len(m.`, fieldname, `)*8`)
  218. } else if repeated {
  219. p.P(`n+=`, strconv.Itoa(key+8), `*len(m.`, fieldname, `)`)
  220. } else if proto3 {
  221. p.P(`if m.`, fieldname, ` != 0 {`)
  222. p.In()
  223. p.P(`n+=`, strconv.Itoa(key+8))
  224. p.Out()
  225. p.P(`}`)
  226. } else if nullable {
  227. p.P(`n+=`, strconv.Itoa(key+8))
  228. } else {
  229. p.P(`n+=`, strconv.Itoa(key+8))
  230. }
  231. case descriptor.FieldDescriptorProto_TYPE_FLOAT,
  232. descriptor.FieldDescriptorProto_TYPE_FIXED32,
  233. descriptor.FieldDescriptorProto_TYPE_SFIXED32:
  234. if packed {
  235. p.P(`n+=`, strconv.Itoa(key), `+sov`, p.localName, `(uint64(len(m.`, fieldname, `)*4))`, `+len(m.`, fieldname, `)*4`)
  236. } else if repeated {
  237. p.P(`n+=`, strconv.Itoa(key+4), `*len(m.`, fieldname, `)`)
  238. } else if proto3 {
  239. p.P(`if m.`, fieldname, ` != 0 {`)
  240. p.In()
  241. p.P(`n+=`, strconv.Itoa(key+4))
  242. p.Out()
  243. p.P(`}`)
  244. } else if nullable {
  245. p.P(`n+=`, strconv.Itoa(key+4))
  246. } else {
  247. p.P(`n+=`, strconv.Itoa(key+4))
  248. }
  249. case descriptor.FieldDescriptorProto_TYPE_INT64,
  250. descriptor.FieldDescriptorProto_TYPE_UINT64,
  251. descriptor.FieldDescriptorProto_TYPE_UINT32,
  252. descriptor.FieldDescriptorProto_TYPE_ENUM,
  253. descriptor.FieldDescriptorProto_TYPE_INT32:
  254. if packed {
  255. p.P(`l = 0`)
  256. p.P(`for _, e := range m.`, fieldname, ` {`)
  257. p.In()
  258. p.P(`l+=sov`, p.localName, `(uint64(e))`)
  259. p.Out()
  260. p.P(`}`)
  261. p.P(`n+=`, strconv.Itoa(key), `+sov`, p.localName, `(uint64(l))+l`)
  262. } else if repeated {
  263. p.P(`for _, e := range m.`, fieldname, ` {`)
  264. p.In()
  265. p.P(`n+=`, strconv.Itoa(key), `+sov`, p.localName, `(uint64(e))`)
  266. p.Out()
  267. p.P(`}`)
  268. } else if proto3 {
  269. p.P(`if m.`, fieldname, ` != 0 {`)
  270. p.In()
  271. p.P(`n+=`, strconv.Itoa(key), `+sov`, p.localName, `(uint64(m.`, fieldname, `))`)
  272. p.Out()
  273. p.P(`}`)
  274. } else if nullable {
  275. p.P(`n+=`, strconv.Itoa(key), `+sov`, p.localName, `(uint64(*m.`, fieldname, `))`)
  276. } else {
  277. p.P(`n+=`, strconv.Itoa(key), `+sov`, p.localName, `(uint64(m.`, fieldname, `))`)
  278. }
  279. case descriptor.FieldDescriptorProto_TYPE_BOOL:
  280. if packed {
  281. p.P(`n+=`, strconv.Itoa(key), `+sov`, p.localName, `(uint64(len(m.`, fieldname, `)))`, `+len(m.`, fieldname, `)*1`)
  282. } else if repeated {
  283. p.P(`n+=`, strconv.Itoa(key+1), `*len(m.`, fieldname, `)`)
  284. } else if proto3 {
  285. p.P(`if m.`, fieldname, ` {`)
  286. p.In()
  287. p.P(`n+=`, strconv.Itoa(key+1))
  288. p.Out()
  289. p.P(`}`)
  290. } else if nullable {
  291. p.P(`n+=`, strconv.Itoa(key+1))
  292. } else {
  293. p.P(`n+=`, strconv.Itoa(key+1))
  294. }
  295. case descriptor.FieldDescriptorProto_TYPE_STRING:
  296. if repeated {
  297. p.P(`for _, s := range m.`, fieldname, ` { `)
  298. p.In()
  299. p.P(`l = len(s)`)
  300. p.P(`n+=`, strconv.Itoa(key), `+l+sov`, p.localName, `(uint64(l))`)
  301. p.Out()
  302. p.P(`}`)
  303. } else if proto3 {
  304. p.P(`l=len(m.`, fieldname, `)`)
  305. p.P(`if l > 0 {`)
  306. p.In()
  307. p.P(`n+=`, strconv.Itoa(key), `+l+sov`, p.localName, `(uint64(l))`)
  308. p.Out()
  309. p.P(`}`)
  310. } else if nullable {
  311. p.P(`l=len(*m.`, fieldname, `)`)
  312. p.P(`n+=`, strconv.Itoa(key), `+l+sov`, p.localName, `(uint64(l))`)
  313. } else {
  314. p.P(`l=len(m.`, fieldname, `)`)
  315. p.P(`n+=`, strconv.Itoa(key), `+l+sov`, p.localName, `(uint64(l))`)
  316. }
  317. case descriptor.FieldDescriptorProto_TYPE_GROUP:
  318. panic(fmt.Errorf("size does not support group %v", fieldname))
  319. case descriptor.FieldDescriptorProto_TYPE_MESSAGE:
  320. if p.IsMap(field) {
  321. m := p.GoMapType(nil, field)
  322. _, keywire := p.GoType(nil, m.KeyAliasField)
  323. valuegoTyp, _ := p.GoType(nil, m.ValueField)
  324. valuegoAliasTyp, valuewire := p.GoType(nil, m.ValueAliasField)
  325. _, fieldwire := p.GoType(nil, field)
  326. nullable, valuegoTyp, valuegoAliasTyp = generator.GoMapValueTypes(field, m.ValueField, valuegoTyp, valuegoAliasTyp)
  327. fieldKeySize := keySize(field.GetNumber(), wireToType(fieldwire))
  328. keyKeySize := keySize(1, wireToType(keywire))
  329. valueKeySize := keySize(2, wireToType(valuewire))
  330. p.P(`for k, v := range m.`, fieldname, ` { `)
  331. p.In()
  332. p.P(`_ = k`)
  333. p.P(`_ = v`)
  334. sum := []string{strconv.Itoa(keyKeySize)}
  335. switch m.KeyField.GetType() {
  336. case descriptor.FieldDescriptorProto_TYPE_DOUBLE,
  337. descriptor.FieldDescriptorProto_TYPE_FIXED64,
  338. descriptor.FieldDescriptorProto_TYPE_SFIXED64:
  339. sum = append(sum, `8`)
  340. case descriptor.FieldDescriptorProto_TYPE_FLOAT,
  341. descriptor.FieldDescriptorProto_TYPE_FIXED32,
  342. descriptor.FieldDescriptorProto_TYPE_SFIXED32:
  343. sum = append(sum, `4`)
  344. case descriptor.FieldDescriptorProto_TYPE_INT64,
  345. descriptor.FieldDescriptorProto_TYPE_UINT64,
  346. descriptor.FieldDescriptorProto_TYPE_UINT32,
  347. descriptor.FieldDescriptorProto_TYPE_ENUM,
  348. descriptor.FieldDescriptorProto_TYPE_INT32:
  349. sum = append(sum, `sov`+p.localName+`(uint64(k))`)
  350. case descriptor.FieldDescriptorProto_TYPE_BOOL:
  351. sum = append(sum, `1`)
  352. case descriptor.FieldDescriptorProto_TYPE_STRING,
  353. descriptor.FieldDescriptorProto_TYPE_BYTES:
  354. sum = append(sum, `len(k)+sov`+p.localName+`(uint64(len(k)))`)
  355. case descriptor.FieldDescriptorProto_TYPE_SINT32,
  356. descriptor.FieldDescriptorProto_TYPE_SINT64:
  357. sum = append(sum, `soz`+p.localName+`(uint64(k))`)
  358. }
  359. switch m.ValueField.GetType() {
  360. case descriptor.FieldDescriptorProto_TYPE_DOUBLE,
  361. descriptor.FieldDescriptorProto_TYPE_FIXED64,
  362. descriptor.FieldDescriptorProto_TYPE_SFIXED64:
  363. sum = append(sum, strconv.Itoa(valueKeySize))
  364. sum = append(sum, strconv.Itoa(8))
  365. case descriptor.FieldDescriptorProto_TYPE_FLOAT,
  366. descriptor.FieldDescriptorProto_TYPE_FIXED32,
  367. descriptor.FieldDescriptorProto_TYPE_SFIXED32:
  368. sum = append(sum, strconv.Itoa(valueKeySize))
  369. sum = append(sum, strconv.Itoa(4))
  370. case descriptor.FieldDescriptorProto_TYPE_INT64,
  371. descriptor.FieldDescriptorProto_TYPE_UINT64,
  372. descriptor.FieldDescriptorProto_TYPE_UINT32,
  373. descriptor.FieldDescriptorProto_TYPE_ENUM,
  374. descriptor.FieldDescriptorProto_TYPE_INT32:
  375. sum = append(sum, strconv.Itoa(valueKeySize))
  376. sum = append(sum, `sov`+p.localName+`(uint64(v))`)
  377. case descriptor.FieldDescriptorProto_TYPE_BOOL:
  378. sum = append(sum, strconv.Itoa(valueKeySize))
  379. sum = append(sum, `1`)
  380. case descriptor.FieldDescriptorProto_TYPE_STRING:
  381. sum = append(sum, strconv.Itoa(valueKeySize))
  382. sum = append(sum, `len(v)+sov`+p.localName+`(uint64(len(v)))`)
  383. case descriptor.FieldDescriptorProto_TYPE_BYTES:
  384. if gogoproto.IsCustomType(field) {
  385. p.P(`l = 0`)
  386. if nullable {
  387. p.P(`if v != nil {`)
  388. p.In()
  389. }
  390. p.P(`l = v.`, sizeName, `()`)
  391. p.P(`l += `, strconv.Itoa(valueKeySize), ` + sov`+p.localName+`(uint64(l))`)
  392. if nullable {
  393. p.Out()
  394. p.P(`}`)
  395. }
  396. sum = append(sum, `l`)
  397. } else {
  398. p.P(`l = 0`)
  399. if proto3 {
  400. p.P(`if len(v) > 0 {`)
  401. } else {
  402. p.P(`if v != nil {`)
  403. }
  404. p.In()
  405. p.P(`l = `, strconv.Itoa(valueKeySize), ` + len(v)+sov`+p.localName+`(uint64(len(v)))`)
  406. p.Out()
  407. p.P(`}`)
  408. sum = append(sum, `l`)
  409. }
  410. case descriptor.FieldDescriptorProto_TYPE_SINT32,
  411. descriptor.FieldDescriptorProto_TYPE_SINT64:
  412. sum = append(sum, strconv.Itoa(valueKeySize))
  413. sum = append(sum, `soz`+p.localName+`(uint64(v))`)
  414. case descriptor.FieldDescriptorProto_TYPE_MESSAGE:
  415. stdSizeCall, stdOk := p.std(field, "v")
  416. if nullable {
  417. p.P(`l = 0`)
  418. p.P(`if v != nil {`)
  419. p.In()
  420. if stdOk {
  421. p.P(`l = `, stdSizeCall)
  422. } else if valuegoTyp != valuegoAliasTyp {
  423. p.P(`l = ((`, valuegoTyp, `)(v)).`, sizeName, `()`)
  424. } else {
  425. p.P(`l = v.`, sizeName, `()`)
  426. }
  427. p.P(`l += `, strconv.Itoa(valueKeySize), ` + sov`+p.localName+`(uint64(l))`)
  428. p.Out()
  429. p.P(`}`)
  430. sum = append(sum, `l`)
  431. } else {
  432. if stdOk {
  433. p.P(`l = `, stdSizeCall)
  434. } else if valuegoTyp != valuegoAliasTyp {
  435. p.P(`l = ((*`, valuegoTyp, `)(&v)).`, sizeName, `()`)
  436. } else {
  437. p.P(`l = v.`, sizeName, `()`)
  438. }
  439. sum = append(sum, strconv.Itoa(valueKeySize))
  440. sum = append(sum, `l+sov`+p.localName+`(uint64(l))`)
  441. }
  442. }
  443. p.P(`mapEntrySize := `, strings.Join(sum, "+"))
  444. p.P(`n+=mapEntrySize+`, fieldKeySize, `+sov`, p.localName, `(uint64(mapEntrySize))`)
  445. p.Out()
  446. p.P(`}`)
  447. } else if repeated {
  448. p.P(`for _, e := range m.`, fieldname, ` { `)
  449. p.In()
  450. stdSizeCall, stdOk := p.std(field, "e")
  451. if stdOk {
  452. p.P(`l=`, stdSizeCall)
  453. } else {
  454. p.P(`l=e.`, sizeName, `()`)
  455. }
  456. p.P(`n+=`, strconv.Itoa(key), `+l+sov`, p.localName, `(uint64(l))`)
  457. p.Out()
  458. p.P(`}`)
  459. } else {
  460. stdSizeCall, stdOk := p.std(field, "m."+fieldname)
  461. if stdOk {
  462. p.P(`l=`, stdSizeCall)
  463. } else {
  464. p.P(`l=m.`, fieldname, `.`, sizeName, `()`)
  465. }
  466. p.P(`n+=`, strconv.Itoa(key), `+l+sov`, p.localName, `(uint64(l))`)
  467. }
  468. case descriptor.FieldDescriptorProto_TYPE_BYTES:
  469. if !gogoproto.IsCustomType(field) {
  470. if repeated {
  471. p.P(`for _, b := range m.`, fieldname, ` { `)
  472. p.In()
  473. p.P(`l = len(b)`)
  474. p.P(`n+=`, strconv.Itoa(key), `+l+sov`, p.localName, `(uint64(l))`)
  475. p.Out()
  476. p.P(`}`)
  477. } else if proto3 {
  478. p.P(`l=len(m.`, fieldname, `)`)
  479. p.P(`if l > 0 {`)
  480. p.In()
  481. p.P(`n+=`, strconv.Itoa(key), `+l+sov`, p.localName, `(uint64(l))`)
  482. p.Out()
  483. p.P(`}`)
  484. } else {
  485. p.P(`l=len(m.`, fieldname, `)`)
  486. p.P(`n+=`, strconv.Itoa(key), `+l+sov`, p.localName, `(uint64(l))`)
  487. }
  488. } else {
  489. if repeated {
  490. p.P(`for _, e := range m.`, fieldname, ` { `)
  491. p.In()
  492. p.P(`l=e.`, sizeName, `()`)
  493. p.P(`n+=`, strconv.Itoa(key), `+l+sov`, p.localName, `(uint64(l))`)
  494. p.Out()
  495. p.P(`}`)
  496. } else {
  497. p.P(`l=m.`, fieldname, `.`, sizeName, `()`)
  498. p.P(`n+=`, strconv.Itoa(key), `+l+sov`, p.localName, `(uint64(l))`)
  499. }
  500. }
  501. case descriptor.FieldDescriptorProto_TYPE_SINT32,
  502. descriptor.FieldDescriptorProto_TYPE_SINT64:
  503. if packed {
  504. p.P(`l = 0`)
  505. p.P(`for _, e := range m.`, fieldname, ` {`)
  506. p.In()
  507. p.P(`l+=soz`, p.localName, `(uint64(e))`)
  508. p.Out()
  509. p.P(`}`)
  510. p.P(`n+=`, strconv.Itoa(key), `+sov`, p.localName, `(uint64(l))+l`)
  511. } else if repeated {
  512. p.P(`for _, e := range m.`, fieldname, ` {`)
  513. p.In()
  514. p.P(`n+=`, strconv.Itoa(key), `+soz`, p.localName, `(uint64(e))`)
  515. p.Out()
  516. p.P(`}`)
  517. } else if proto3 {
  518. p.P(`if m.`, fieldname, ` != 0 {`)
  519. p.In()
  520. p.P(`n+=`, strconv.Itoa(key), `+soz`, p.localName, `(uint64(m.`, fieldname, `))`)
  521. p.Out()
  522. p.P(`}`)
  523. } else if nullable {
  524. p.P(`n+=`, strconv.Itoa(key), `+soz`, p.localName, `(uint64(*m.`, fieldname, `))`)
  525. } else {
  526. p.P(`n+=`, strconv.Itoa(key), `+soz`, p.localName, `(uint64(m.`, fieldname, `))`)
  527. }
  528. default:
  529. panic("not implemented")
  530. }
  531. if repeated || doNilCheck {
  532. p.Out()
  533. p.P(`}`)
  534. }
  535. }
  536. func (p *size) Generate(file *generator.FileDescriptor) {
  537. p.PluginImports = generator.NewPluginImports(p.Generator)
  538. p.atleastOne = false
  539. p.localName = generator.FileName(file)
  540. p.typesPkg = p.NewImport("github.com/gogo/protobuf/types")
  541. protoPkg := p.NewImport("github.com/gogo/protobuf/proto")
  542. if !gogoproto.ImportsGoGoProto(file.FileDescriptorProto) {
  543. protoPkg = p.NewImport("github.com/golang/protobuf/proto")
  544. }
  545. for _, message := range file.Messages() {
  546. sizeName := ""
  547. if gogoproto.IsSizer(file.FileDescriptorProto, message.DescriptorProto) && gogoproto.IsProtoSizer(file.FileDescriptorProto, message.DescriptorProto) {
  548. fmt.Fprintf(os.Stderr, "ERROR: message %v cannot support both sizer and protosizer plugins\n", generator.CamelCase(*message.Name))
  549. os.Exit(1)
  550. }
  551. if gogoproto.IsSizer(file.FileDescriptorProto, message.DescriptorProto) {
  552. sizeName = "Size"
  553. } else if gogoproto.IsProtoSizer(file.FileDescriptorProto, message.DescriptorProto) {
  554. sizeName = "ProtoSize"
  555. } else {
  556. continue
  557. }
  558. if message.DescriptorProto.GetOptions().GetMapEntry() {
  559. continue
  560. }
  561. p.atleastOne = true
  562. ccTypeName := generator.CamelCaseSlice(message.TypeName())
  563. p.P(`func (m *`, ccTypeName, `) `, sizeName, `() (n int) {`)
  564. p.In()
  565. p.P(`if m == nil {`)
  566. p.In()
  567. p.P(`return 0`)
  568. p.Out()
  569. p.P(`}`)
  570. p.P(`var l int`)
  571. p.P(`_ = l`)
  572. oneofs := make(map[string]struct{})
  573. for _, field := range message.Field {
  574. oneof := field.OneofIndex != nil
  575. if !oneof {
  576. proto3 := gogoproto.IsProto3(file.FileDescriptorProto)
  577. p.generateField(proto3, file, message, field, sizeName)
  578. } else {
  579. fieldname := p.GetFieldName(message, field)
  580. if _, ok := oneofs[fieldname]; ok {
  581. continue
  582. } else {
  583. oneofs[fieldname] = struct{}{}
  584. }
  585. p.P(`if m.`, fieldname, ` != nil {`)
  586. p.In()
  587. p.P(`n+=m.`, fieldname, `.`, sizeName, `()`)
  588. p.Out()
  589. p.P(`}`)
  590. }
  591. }
  592. if message.DescriptorProto.HasExtension() {
  593. if gogoproto.HasExtensionsMap(file.FileDescriptorProto, message.DescriptorProto) {
  594. p.P(`n += `, protoPkg.Use(), `.SizeOfInternalExtension(m)`)
  595. } else {
  596. p.P(`if m.XXX_extensions != nil {`)
  597. p.In()
  598. p.P(`n+=len(m.XXX_extensions)`)
  599. p.Out()
  600. p.P(`}`)
  601. }
  602. }
  603. if gogoproto.HasUnrecognized(file.FileDescriptorProto, message.DescriptorProto) {
  604. p.P(`if m.XXX_unrecognized != nil {`)
  605. p.In()
  606. p.P(`n+=len(m.XXX_unrecognized)`)
  607. p.Out()
  608. p.P(`}`)
  609. }
  610. p.P(`return n`)
  611. p.Out()
  612. p.P(`}`)
  613. p.P()
  614. //Generate Size methods for oneof fields
  615. m := proto.Clone(message.DescriptorProto).(*descriptor.DescriptorProto)
  616. for _, f := range m.Field {
  617. oneof := f.OneofIndex != nil
  618. if !oneof {
  619. continue
  620. }
  621. ccTypeName := p.OneOfTypeName(message, f)
  622. p.P(`func (m *`, ccTypeName, `) `, sizeName, `() (n int) {`)
  623. p.In()
  624. p.P(`if m == nil {`)
  625. p.In()
  626. p.P(`return 0`)
  627. p.Out()
  628. p.P(`}`)
  629. p.P(`var l int`)
  630. p.P(`_ = l`)
  631. vanity.TurnOffNullableForNativeTypes(f)
  632. p.generateField(false, file, message, f, sizeName)
  633. p.P(`return n`)
  634. p.Out()
  635. p.P(`}`)
  636. }
  637. }
  638. if !p.atleastOne {
  639. return
  640. }
  641. p.sizeVarint()
  642. p.sizeZigZag()
  643. }
  644. func init() {
  645. generator.RegisterPlugin(NewSize())
  646. }