marshalto.go 36 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205
  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 marshalto plugin generates a Marshal and MarshalTo method for each message.
  30. The `Marshal() ([]byte, error)` method results in the fact that the message
  31. implements the Marshaler interface.
  32. This allows proto.Marshal to be faster by calling the generated Marshal method rather than using reflect to Marshal the struct.
  33. If is enabled by the following extensions:
  34. - marshaler
  35. - marshaler_all
  36. Or the following extensions:
  37. - unsafe_marshaler
  38. - unsafe_marshaler_all
  39. That is if you want to use the unsafe package in your generated code.
  40. The speed up using the unsafe package is not very significant.
  41. The generation of marshalling tests are enabled using one of the following extensions:
  42. - testgen
  43. - testgen_all
  44. And benchmarks given it is enabled using one of the following extensions:
  45. - benchgen
  46. - benchgen_all
  47. Let us look at:
  48. github.com/gogo/protobuf/test/example/example.proto
  49. Btw all the output can be seen at:
  50. github.com/gogo/protobuf/test/example/*
  51. The following message:
  52. option (gogoproto.marshaler_all) = true;
  53. message B {
  54. option (gogoproto.description) = true;
  55. optional A A = 1 [(gogoproto.nullable) = false, (gogoproto.embed) = true];
  56. repeated bytes G = 2 [(gogoproto.customtype) = "github.com/gogo/protobuf/test/custom.Uint128", (gogoproto.nullable) = false];
  57. }
  58. given to the marshalto plugin, will generate the following code:
  59. func (m *B) Marshal() (dAtA []byte, err error) {
  60. size := m.Size()
  61. dAtA = make([]byte, size)
  62. n, err := m.MarshalTo(dAtA)
  63. if err != nil {
  64. return nil, err
  65. }
  66. return dAtA[:n], nil
  67. }
  68. func (m *B) MarshalTo(dAtA []byte) (int, error) {
  69. var i int
  70. _ = i
  71. var l int
  72. _ = l
  73. dAtA[i] = 0xa
  74. i++
  75. i = encodeVarintExample(dAtA, i, uint64(m.A.Size()))
  76. n2, err := m.A.MarshalTo(dAtA[i:])
  77. if err != nil {
  78. return 0, err
  79. }
  80. i += n2
  81. if len(m.G) > 0 {
  82. for _, msg := range m.G {
  83. dAtA[i] = 0x12
  84. i++
  85. i = encodeVarintExample(dAtA, i, uint64(msg.Size()))
  86. n, err := msg.MarshalTo(dAtA[i:])
  87. if err != nil {
  88. return 0, err
  89. }
  90. i += n
  91. }
  92. }
  93. if m.XXX_unrecognized != nil {
  94. i += copy(dAtA[i:], m.XXX_unrecognized)
  95. }
  96. return i, nil
  97. }
  98. As shown above Marshal calculates the size of the not yet marshalled message
  99. and allocates the appropriate buffer.
  100. This is followed by calling the MarshalTo method which requires a preallocated buffer.
  101. The MarshalTo method allows a user to rather preallocated a reusable buffer.
  102. The Size method is generated using the size plugin and the gogoproto.sizer, gogoproto.sizer_all extensions.
  103. The user can also using the generated Size method to check that his reusable buffer is still big enough.
  104. The generated tests and benchmarks will keep you safe and show that this is really a significant speed improvement.
  105. An additional message-level option `stable_marshaler` (and the file-level
  106. option `stable_marshaler_all`) exists which causes the generated marshalling
  107. code to behave deterministically. Today, this only changes the serialization of
  108. maps; they are serialized in sort order.
  109. */
  110. package marshalto
  111. import (
  112. "fmt"
  113. "sort"
  114. "strconv"
  115. "strings"
  116. "github.com/gogo/protobuf/gogoproto"
  117. "github.com/gogo/protobuf/proto"
  118. descriptor "github.com/gogo/protobuf/protoc-gen-gogo/descriptor"
  119. "github.com/gogo/protobuf/protoc-gen-gogo/generator"
  120. "github.com/gogo/protobuf/vanity"
  121. )
  122. type NumGen interface {
  123. Next() string
  124. Current() string
  125. }
  126. type numGen struct {
  127. index int
  128. }
  129. func NewNumGen() NumGen {
  130. return &numGen{0}
  131. }
  132. func (this *numGen) Next() string {
  133. this.index++
  134. return this.Current()
  135. }
  136. func (this *numGen) Current() string {
  137. return strconv.Itoa(this.index)
  138. }
  139. type marshalto struct {
  140. *generator.Generator
  141. generator.PluginImports
  142. atleastOne bool
  143. errorsPkg generator.Single
  144. protoPkg generator.Single
  145. sortKeysPkg generator.Single
  146. mathPkg generator.Single
  147. typesPkg generator.Single
  148. binaryPkg generator.Single
  149. localName string
  150. }
  151. func NewMarshal() *marshalto {
  152. return &marshalto{}
  153. }
  154. func (p *marshalto) Name() string {
  155. return "marshalto"
  156. }
  157. func (p *marshalto) Init(g *generator.Generator) {
  158. p.Generator = g
  159. }
  160. func (p *marshalto) callFixed64(varName ...string) {
  161. p.P(p.binaryPkg.Use(), `.LittleEndian.PutUint64(dAtA[i:], uint64(`, strings.Join(varName, ""), `))`)
  162. p.P(`i += 8`)
  163. }
  164. func (p *marshalto) callFixed32(varName ...string) {
  165. p.P(p.binaryPkg.Use(), `.LittleEndian.PutUint32(dAtA[i:], uint32(`, strings.Join(varName, ""), `))`)
  166. p.P(`i += 4`)
  167. }
  168. func (p *marshalto) callVarint(varName ...string) {
  169. p.P(`i = encodeVarint`, p.localName, `(dAtA, i, uint64(`, strings.Join(varName, ""), `))`)
  170. }
  171. func (p *marshalto) encodeVarint(varName string) {
  172. p.P(`for `, varName, ` >= 1<<7 {`)
  173. p.In()
  174. p.P(`dAtA[i] = uint8(uint64(`, varName, `)&0x7f|0x80)`)
  175. p.P(varName, ` >>= 7`)
  176. p.P(`i++`)
  177. p.Out()
  178. p.P(`}`)
  179. p.P(`dAtA[i] = uint8(`, varName, `)`)
  180. p.P(`i++`)
  181. }
  182. func (p *marshalto) encodeKey(fieldNumber int32, wireType int) {
  183. x := uint32(fieldNumber)<<3 | uint32(wireType)
  184. i := 0
  185. keybuf := make([]byte, 0)
  186. for i = 0; x > 127; i++ {
  187. keybuf = append(keybuf, 0x80|uint8(x&0x7F))
  188. x >>= 7
  189. }
  190. keybuf = append(keybuf, uint8(x))
  191. for _, b := range keybuf {
  192. p.P(`dAtA[i] = `, fmt.Sprintf("%#v", b))
  193. p.P(`i++`)
  194. }
  195. }
  196. func keySize(fieldNumber int32, wireType int) int {
  197. x := uint32(fieldNumber)<<3 | uint32(wireType)
  198. size := 0
  199. for size = 0; x > 127; size++ {
  200. x >>= 7
  201. }
  202. size++
  203. return size
  204. }
  205. func wireToType(wire string) int {
  206. switch wire {
  207. case "fixed64":
  208. return proto.WireFixed64
  209. case "fixed32":
  210. return proto.WireFixed32
  211. case "varint":
  212. return proto.WireVarint
  213. case "bytes":
  214. return proto.WireBytes
  215. case "group":
  216. return proto.WireBytes
  217. case "zigzag32":
  218. return proto.WireVarint
  219. case "zigzag64":
  220. return proto.WireVarint
  221. }
  222. panic("unreachable")
  223. }
  224. func (p *marshalto) mapField(numGen NumGen, field *descriptor.FieldDescriptorProto, kvField *descriptor.FieldDescriptorProto, varName string, protoSizer bool) {
  225. switch kvField.GetType() {
  226. case descriptor.FieldDescriptorProto_TYPE_DOUBLE:
  227. p.callFixed64(p.mathPkg.Use(), `.Float64bits(float64(`, varName, `))`)
  228. case descriptor.FieldDescriptorProto_TYPE_FLOAT:
  229. p.callFixed32(p.mathPkg.Use(), `.Float32bits(float32(`, varName, `))`)
  230. case descriptor.FieldDescriptorProto_TYPE_INT64,
  231. descriptor.FieldDescriptorProto_TYPE_UINT64,
  232. descriptor.FieldDescriptorProto_TYPE_INT32,
  233. descriptor.FieldDescriptorProto_TYPE_UINT32,
  234. descriptor.FieldDescriptorProto_TYPE_ENUM:
  235. p.callVarint(varName)
  236. case descriptor.FieldDescriptorProto_TYPE_FIXED64,
  237. descriptor.FieldDescriptorProto_TYPE_SFIXED64:
  238. p.callFixed64(varName)
  239. case descriptor.FieldDescriptorProto_TYPE_FIXED32,
  240. descriptor.FieldDescriptorProto_TYPE_SFIXED32:
  241. p.callFixed32(varName)
  242. case descriptor.FieldDescriptorProto_TYPE_BOOL:
  243. p.P(`if `, varName, ` {`)
  244. p.In()
  245. p.P(`dAtA[i] = 1`)
  246. p.Out()
  247. p.P(`} else {`)
  248. p.In()
  249. p.P(`dAtA[i] = 0`)
  250. p.Out()
  251. p.P(`}`)
  252. p.P(`i++`)
  253. case descriptor.FieldDescriptorProto_TYPE_STRING,
  254. descriptor.FieldDescriptorProto_TYPE_BYTES:
  255. if gogoproto.IsCustomType(field) && kvField.IsBytes() {
  256. p.callVarint(varName, `.Size()`)
  257. p.P(`n`, numGen.Next(), `, err := `, varName, `.MarshalTo(dAtA[i:])`)
  258. p.P(`if err != nil {`)
  259. p.In()
  260. p.P(`return 0, err`)
  261. p.Out()
  262. p.P(`}`)
  263. p.P(`i+=n`, numGen.Current())
  264. } else {
  265. p.callVarint(`len(`, varName, `)`)
  266. p.P(`i+=copy(dAtA[i:], `, varName, `)`)
  267. }
  268. case descriptor.FieldDescriptorProto_TYPE_SINT32:
  269. p.callVarint(`(uint32(`, varName, `) << 1) ^ uint32((`, varName, ` >> 31))`)
  270. case descriptor.FieldDescriptorProto_TYPE_SINT64:
  271. p.callVarint(`(uint64(`, varName, `) << 1) ^ uint64((`, varName, ` >> 63))`)
  272. case descriptor.FieldDescriptorProto_TYPE_MESSAGE:
  273. if gogoproto.IsStdTime(field) {
  274. p.callVarint(p.typesPkg.Use(), `.SizeOfStdTime(*`, varName, `)`)
  275. p.P(`n`, numGen.Next(), `, err := `, p.typesPkg.Use(), `.StdTimeMarshalTo(*`, varName, `, dAtA[i:])`)
  276. } else if gogoproto.IsStdDuration(field) {
  277. p.callVarint(p.typesPkg.Use(), `.SizeOfStdDuration(*`, varName, `)`)
  278. p.P(`n`, numGen.Next(), `, err := `, p.typesPkg.Use(), `.StdDurationMarshalTo(*`, varName, `, dAtA[i:])`)
  279. } else if protoSizer {
  280. p.callVarint(varName, `.ProtoSize()`)
  281. p.P(`n`, numGen.Next(), `, err := `, varName, `.MarshalTo(dAtA[i:])`)
  282. } else {
  283. p.callVarint(varName, `.Size()`)
  284. p.P(`n`, numGen.Next(), `, err := `, varName, `.MarshalTo(dAtA[i:])`)
  285. }
  286. p.P(`if err != nil {`)
  287. p.In()
  288. p.P(`return 0, err`)
  289. p.Out()
  290. p.P(`}`)
  291. p.P(`i+=n`, numGen.Current())
  292. }
  293. }
  294. type orderFields []*descriptor.FieldDescriptorProto
  295. func (this orderFields) Len() int {
  296. return len(this)
  297. }
  298. func (this orderFields) Less(i, j int) bool {
  299. return this[i].GetNumber() < this[j].GetNumber()
  300. }
  301. func (this orderFields) Swap(i, j int) {
  302. this[i], this[j] = this[j], this[i]
  303. }
  304. func (p *marshalto) generateField(proto3 bool, numGen NumGen, file *generator.FileDescriptor, message *generator.Descriptor, field *descriptor.FieldDescriptorProto) {
  305. fieldname := p.GetOneOfFieldName(message, field)
  306. nullable := gogoproto.IsNullable(field)
  307. repeated := field.IsRepeated()
  308. required := field.IsRequired()
  309. protoSizer := gogoproto.IsProtoSizer(file.FileDescriptorProto, message.DescriptorProto)
  310. doNilCheck := gogoproto.NeedsNilCheck(proto3, field)
  311. if required && nullable {
  312. p.P(`if m.`, fieldname, `== nil {`)
  313. p.In()
  314. if !gogoproto.ImportsGoGoProto(file.FileDescriptorProto) {
  315. p.P(`return 0, new(`, p.protoPkg.Use(), `.RequiredNotSetError)`)
  316. } else {
  317. p.P(`return 0, `, p.protoPkg.Use(), `.NewRequiredNotSetError("`, field.GetName(), `")`)
  318. }
  319. p.Out()
  320. p.P(`} else {`)
  321. } else if repeated {
  322. p.P(`if len(m.`, fieldname, `) > 0 {`)
  323. p.In()
  324. } else if doNilCheck {
  325. p.P(`if m.`, fieldname, ` != nil {`)
  326. p.In()
  327. }
  328. packed := field.IsPacked() || (proto3 && field.IsPacked3())
  329. wireType := field.WireType()
  330. fieldNumber := field.GetNumber()
  331. if packed {
  332. wireType = proto.WireBytes
  333. }
  334. switch *field.Type {
  335. case descriptor.FieldDescriptorProto_TYPE_DOUBLE:
  336. if packed {
  337. p.encodeKey(fieldNumber, wireType)
  338. p.callVarint(`len(m.`, fieldname, `) * 8`)
  339. p.P(`for _, num := range m.`, fieldname, ` {`)
  340. p.In()
  341. p.P(`f`, numGen.Next(), ` := `, p.mathPkg.Use(), `.Float64bits(float64(num))`)
  342. p.callFixed64("f" + numGen.Current())
  343. p.Out()
  344. p.P(`}`)
  345. } else if repeated {
  346. p.P(`for _, num := range m.`, fieldname, ` {`)
  347. p.In()
  348. p.encodeKey(fieldNumber, wireType)
  349. p.P(`f`, numGen.Next(), ` := `, p.mathPkg.Use(), `.Float64bits(float64(num))`)
  350. p.callFixed64("f" + numGen.Current())
  351. p.Out()
  352. p.P(`}`)
  353. } else if proto3 {
  354. p.P(`if m.`, fieldname, ` != 0 {`)
  355. p.In()
  356. p.encodeKey(fieldNumber, wireType)
  357. p.callFixed64(p.mathPkg.Use(), `.Float64bits(float64(m.`+fieldname, `))`)
  358. p.Out()
  359. p.P(`}`)
  360. } else if !nullable {
  361. p.encodeKey(fieldNumber, wireType)
  362. p.callFixed64(p.mathPkg.Use(), `.Float64bits(float64(m.`+fieldname, `))`)
  363. } else {
  364. p.encodeKey(fieldNumber, wireType)
  365. p.callFixed64(p.mathPkg.Use(), `.Float64bits(float64(*m.`+fieldname, `))`)
  366. }
  367. case descriptor.FieldDescriptorProto_TYPE_FLOAT:
  368. if packed {
  369. p.encodeKey(fieldNumber, wireType)
  370. p.callVarint(`len(m.`, fieldname, `) * 4`)
  371. p.P(`for _, num := range m.`, fieldname, ` {`)
  372. p.In()
  373. p.P(`f`, numGen.Next(), ` := `, p.mathPkg.Use(), `.Float32bits(float32(num))`)
  374. p.callFixed32("f" + numGen.Current())
  375. p.Out()
  376. p.P(`}`)
  377. } else if repeated {
  378. p.P(`for _, num := range m.`, fieldname, ` {`)
  379. p.In()
  380. p.encodeKey(fieldNumber, wireType)
  381. p.P(`f`, numGen.Next(), ` := `, p.mathPkg.Use(), `.Float32bits(float32(num))`)
  382. p.callFixed32("f" + numGen.Current())
  383. p.Out()
  384. p.P(`}`)
  385. } else if proto3 {
  386. p.P(`if m.`, fieldname, ` != 0 {`)
  387. p.In()
  388. p.encodeKey(fieldNumber, wireType)
  389. p.callFixed32(p.mathPkg.Use(), `.Float32bits(float32(m.`+fieldname, `))`)
  390. p.Out()
  391. p.P(`}`)
  392. } else if !nullable {
  393. p.encodeKey(fieldNumber, wireType)
  394. p.callFixed32(p.mathPkg.Use(), `.Float32bits(float32(m.`+fieldname, `))`)
  395. } else {
  396. p.encodeKey(fieldNumber, wireType)
  397. p.callFixed32(p.mathPkg.Use(), `.Float32bits(float32(*m.`+fieldname, `))`)
  398. }
  399. case descriptor.FieldDescriptorProto_TYPE_INT64,
  400. descriptor.FieldDescriptorProto_TYPE_UINT64,
  401. descriptor.FieldDescriptorProto_TYPE_INT32,
  402. descriptor.FieldDescriptorProto_TYPE_UINT32,
  403. descriptor.FieldDescriptorProto_TYPE_ENUM:
  404. if packed {
  405. jvar := "j" + numGen.Next()
  406. p.P(`dAtA`, numGen.Next(), ` := make([]byte, len(m.`, fieldname, `)*10)`)
  407. p.P(`var `, jvar, ` int`)
  408. if *field.Type == descriptor.FieldDescriptorProto_TYPE_INT64 ||
  409. *field.Type == descriptor.FieldDescriptorProto_TYPE_INT32 {
  410. p.P(`for _, num1 := range m.`, fieldname, ` {`)
  411. p.In()
  412. p.P(`num := uint64(num1)`)
  413. } else {
  414. p.P(`for _, num := range m.`, fieldname, ` {`)
  415. p.In()
  416. }
  417. p.P(`for num >= 1<<7 {`)
  418. p.In()
  419. p.P(`dAtA`, numGen.Current(), `[`, jvar, `] = uint8(uint64(num)&0x7f|0x80)`)
  420. p.P(`num >>= 7`)
  421. p.P(jvar, `++`)
  422. p.Out()
  423. p.P(`}`)
  424. p.P(`dAtA`, numGen.Current(), `[`, jvar, `] = uint8(num)`)
  425. p.P(jvar, `++`)
  426. p.Out()
  427. p.P(`}`)
  428. p.encodeKey(fieldNumber, wireType)
  429. p.callVarint(jvar)
  430. p.P(`i += copy(dAtA[i:], dAtA`, numGen.Current(), `[:`, jvar, `])`)
  431. } else if repeated {
  432. p.P(`for _, num := range m.`, fieldname, ` {`)
  433. p.In()
  434. p.encodeKey(fieldNumber, wireType)
  435. p.callVarint("num")
  436. p.Out()
  437. p.P(`}`)
  438. } else if proto3 {
  439. p.P(`if m.`, fieldname, ` != 0 {`)
  440. p.In()
  441. p.encodeKey(fieldNumber, wireType)
  442. p.callVarint(`m.`, fieldname)
  443. p.Out()
  444. p.P(`}`)
  445. } else if !nullable {
  446. p.encodeKey(fieldNumber, wireType)
  447. p.callVarint(`m.`, fieldname)
  448. } else {
  449. p.encodeKey(fieldNumber, wireType)
  450. p.callVarint(`*m.`, fieldname)
  451. }
  452. case descriptor.FieldDescriptorProto_TYPE_FIXED64,
  453. descriptor.FieldDescriptorProto_TYPE_SFIXED64:
  454. if packed {
  455. p.encodeKey(fieldNumber, wireType)
  456. p.callVarint(`len(m.`, fieldname, `) * 8`)
  457. p.P(`for _, num := range m.`, fieldname, ` {`)
  458. p.In()
  459. p.callFixed64("num")
  460. p.Out()
  461. p.P(`}`)
  462. } else if repeated {
  463. p.P(`for _, num := range m.`, fieldname, ` {`)
  464. p.In()
  465. p.encodeKey(fieldNumber, wireType)
  466. p.callFixed64("num")
  467. p.Out()
  468. p.P(`}`)
  469. } else if proto3 {
  470. p.P(`if m.`, fieldname, ` != 0 {`)
  471. p.In()
  472. p.encodeKey(fieldNumber, wireType)
  473. p.callFixed64("m." + fieldname)
  474. p.Out()
  475. p.P(`}`)
  476. } else if !nullable {
  477. p.encodeKey(fieldNumber, wireType)
  478. p.callFixed64("m." + fieldname)
  479. } else {
  480. p.encodeKey(fieldNumber, wireType)
  481. p.callFixed64("*m." + fieldname)
  482. }
  483. case descriptor.FieldDescriptorProto_TYPE_FIXED32,
  484. descriptor.FieldDescriptorProto_TYPE_SFIXED32:
  485. if packed {
  486. p.encodeKey(fieldNumber, wireType)
  487. p.callVarint(`len(m.`, fieldname, `) * 4`)
  488. p.P(`for _, num := range m.`, fieldname, ` {`)
  489. p.In()
  490. p.callFixed32("num")
  491. p.Out()
  492. p.P(`}`)
  493. } else if repeated {
  494. p.P(`for _, num := range m.`, fieldname, ` {`)
  495. p.In()
  496. p.encodeKey(fieldNumber, wireType)
  497. p.callFixed32("num")
  498. p.Out()
  499. p.P(`}`)
  500. } else if proto3 {
  501. p.P(`if m.`, fieldname, ` != 0 {`)
  502. p.In()
  503. p.encodeKey(fieldNumber, wireType)
  504. p.callFixed32("m." + fieldname)
  505. p.Out()
  506. p.P(`}`)
  507. } else if !nullable {
  508. p.encodeKey(fieldNumber, wireType)
  509. p.callFixed32("m." + fieldname)
  510. } else {
  511. p.encodeKey(fieldNumber, wireType)
  512. p.callFixed32("*m." + fieldname)
  513. }
  514. case descriptor.FieldDescriptorProto_TYPE_BOOL:
  515. if packed {
  516. p.encodeKey(fieldNumber, wireType)
  517. p.callVarint(`len(m.`, fieldname, `)`)
  518. p.P(`for _, b := range m.`, fieldname, ` {`)
  519. p.In()
  520. p.P(`if b {`)
  521. p.In()
  522. p.P(`dAtA[i] = 1`)
  523. p.Out()
  524. p.P(`} else {`)
  525. p.In()
  526. p.P(`dAtA[i] = 0`)
  527. p.Out()
  528. p.P(`}`)
  529. p.P(`i++`)
  530. p.Out()
  531. p.P(`}`)
  532. } else if repeated {
  533. p.P(`for _, b := range m.`, fieldname, ` {`)
  534. p.In()
  535. p.encodeKey(fieldNumber, wireType)
  536. p.P(`if b {`)
  537. p.In()
  538. p.P(`dAtA[i] = 1`)
  539. p.Out()
  540. p.P(`} else {`)
  541. p.In()
  542. p.P(`dAtA[i] = 0`)
  543. p.Out()
  544. p.P(`}`)
  545. p.P(`i++`)
  546. p.Out()
  547. p.P(`}`)
  548. } else if proto3 {
  549. p.P(`if m.`, fieldname, ` {`)
  550. p.In()
  551. p.encodeKey(fieldNumber, wireType)
  552. p.P(`if m.`, fieldname, ` {`)
  553. p.In()
  554. p.P(`dAtA[i] = 1`)
  555. p.Out()
  556. p.P(`} else {`)
  557. p.In()
  558. p.P(`dAtA[i] = 0`)
  559. p.Out()
  560. p.P(`}`)
  561. p.P(`i++`)
  562. p.Out()
  563. p.P(`}`)
  564. } else if !nullable {
  565. p.encodeKey(fieldNumber, wireType)
  566. p.P(`if m.`, fieldname, ` {`)
  567. p.In()
  568. p.P(`dAtA[i] = 1`)
  569. p.Out()
  570. p.P(`} else {`)
  571. p.In()
  572. p.P(`dAtA[i] = 0`)
  573. p.Out()
  574. p.P(`}`)
  575. p.P(`i++`)
  576. } else {
  577. p.encodeKey(fieldNumber, wireType)
  578. p.P(`if *m.`, fieldname, ` {`)
  579. p.In()
  580. p.P(`dAtA[i] = 1`)
  581. p.Out()
  582. p.P(`} else {`)
  583. p.In()
  584. p.P(`dAtA[i] = 0`)
  585. p.Out()
  586. p.P(`}`)
  587. p.P(`i++`)
  588. }
  589. case descriptor.FieldDescriptorProto_TYPE_STRING:
  590. if repeated {
  591. p.P(`for _, s := range m.`, fieldname, ` {`)
  592. p.In()
  593. p.encodeKey(fieldNumber, wireType)
  594. p.P(`l = len(s)`)
  595. p.encodeVarint("l")
  596. p.P(`i+=copy(dAtA[i:], s)`)
  597. p.Out()
  598. p.P(`}`)
  599. } else if proto3 {
  600. p.P(`if len(m.`, fieldname, `) > 0 {`)
  601. p.In()
  602. p.encodeKey(fieldNumber, wireType)
  603. p.callVarint(`len(m.`, fieldname, `)`)
  604. p.P(`i+=copy(dAtA[i:], m.`, fieldname, `)`)
  605. p.Out()
  606. p.P(`}`)
  607. } else if !nullable {
  608. p.encodeKey(fieldNumber, wireType)
  609. p.callVarint(`len(m.`, fieldname, `)`)
  610. p.P(`i+=copy(dAtA[i:], m.`, fieldname, `)`)
  611. } else {
  612. p.encodeKey(fieldNumber, wireType)
  613. p.callVarint(`len(*m.`, fieldname, `)`)
  614. p.P(`i+=copy(dAtA[i:], *m.`, fieldname, `)`)
  615. }
  616. case descriptor.FieldDescriptorProto_TYPE_GROUP:
  617. panic(fmt.Errorf("marshaler does not support group %v", fieldname))
  618. case descriptor.FieldDescriptorProto_TYPE_MESSAGE:
  619. if p.IsMap(field) {
  620. m := p.GoMapType(nil, field)
  621. keygoTyp, keywire := p.GoType(nil, m.KeyField)
  622. keygoAliasTyp, _ := p.GoType(nil, m.KeyAliasField)
  623. // keys may not be pointers
  624. keygoTyp = strings.Replace(keygoTyp, "*", "", 1)
  625. keygoAliasTyp = strings.Replace(keygoAliasTyp, "*", "", 1)
  626. keyCapTyp := generator.CamelCase(keygoTyp)
  627. valuegoTyp, valuewire := p.GoType(nil, m.ValueField)
  628. valuegoAliasTyp, _ := p.GoType(nil, m.ValueAliasField)
  629. nullable, valuegoTyp, valuegoAliasTyp = generator.GoMapValueTypes(field, m.ValueField, valuegoTyp, valuegoAliasTyp)
  630. keyKeySize := keySize(1, wireToType(keywire))
  631. valueKeySize := keySize(2, wireToType(valuewire))
  632. if gogoproto.IsStableMarshaler(file.FileDescriptorProto, message.DescriptorProto) {
  633. keysName := `keysFor` + fieldname
  634. p.P(keysName, ` := make([]`, keygoTyp, `, 0, len(m.`, fieldname, `))`)
  635. p.P(`for k, _ := range m.`, fieldname, ` {`)
  636. p.In()
  637. p.P(keysName, ` = append(`, keysName, `, `, keygoTyp, `(k))`)
  638. p.Out()
  639. p.P(`}`)
  640. p.P(p.sortKeysPkg.Use(), `.`, keyCapTyp, `s(`, keysName, `)`)
  641. p.P(`for _, k := range `, keysName, ` {`)
  642. } else {
  643. p.P(`for k, _ := range m.`, fieldname, ` {`)
  644. }
  645. p.In()
  646. p.encodeKey(fieldNumber, wireType)
  647. sum := []string{strconv.Itoa(keyKeySize)}
  648. switch m.KeyField.GetType() {
  649. case descriptor.FieldDescriptorProto_TYPE_DOUBLE,
  650. descriptor.FieldDescriptorProto_TYPE_FIXED64,
  651. descriptor.FieldDescriptorProto_TYPE_SFIXED64:
  652. sum = append(sum, `8`)
  653. case descriptor.FieldDescriptorProto_TYPE_FLOAT,
  654. descriptor.FieldDescriptorProto_TYPE_FIXED32,
  655. descriptor.FieldDescriptorProto_TYPE_SFIXED32:
  656. sum = append(sum, `4`)
  657. case descriptor.FieldDescriptorProto_TYPE_INT64,
  658. descriptor.FieldDescriptorProto_TYPE_UINT64,
  659. descriptor.FieldDescriptorProto_TYPE_UINT32,
  660. descriptor.FieldDescriptorProto_TYPE_ENUM,
  661. descriptor.FieldDescriptorProto_TYPE_INT32:
  662. sum = append(sum, `sov`+p.localName+`(uint64(k))`)
  663. case descriptor.FieldDescriptorProto_TYPE_BOOL:
  664. sum = append(sum, `1`)
  665. case descriptor.FieldDescriptorProto_TYPE_STRING,
  666. descriptor.FieldDescriptorProto_TYPE_BYTES:
  667. sum = append(sum, `len(k)+sov`+p.localName+`(uint64(len(k)))`)
  668. case descriptor.FieldDescriptorProto_TYPE_SINT32,
  669. descriptor.FieldDescriptorProto_TYPE_SINT64:
  670. sum = append(sum, `soz`+p.localName+`(uint64(k))`)
  671. }
  672. if gogoproto.IsStableMarshaler(file.FileDescriptorProto, message.DescriptorProto) {
  673. p.P(`v := m.`, fieldname, `[`, keygoAliasTyp, `(k)]`)
  674. } else {
  675. p.P(`v := m.`, fieldname, `[k]`)
  676. }
  677. accessor := `v`
  678. switch m.ValueField.GetType() {
  679. case descriptor.FieldDescriptorProto_TYPE_DOUBLE,
  680. descriptor.FieldDescriptorProto_TYPE_FIXED64,
  681. descriptor.FieldDescriptorProto_TYPE_SFIXED64:
  682. sum = append(sum, strconv.Itoa(valueKeySize))
  683. sum = append(sum, strconv.Itoa(8))
  684. case descriptor.FieldDescriptorProto_TYPE_FLOAT,
  685. descriptor.FieldDescriptorProto_TYPE_FIXED32,
  686. descriptor.FieldDescriptorProto_TYPE_SFIXED32:
  687. sum = append(sum, strconv.Itoa(valueKeySize))
  688. sum = append(sum, strconv.Itoa(4))
  689. case descriptor.FieldDescriptorProto_TYPE_INT64,
  690. descriptor.FieldDescriptorProto_TYPE_UINT64,
  691. descriptor.FieldDescriptorProto_TYPE_UINT32,
  692. descriptor.FieldDescriptorProto_TYPE_ENUM,
  693. descriptor.FieldDescriptorProto_TYPE_INT32:
  694. sum = append(sum, strconv.Itoa(valueKeySize))
  695. sum = append(sum, `sov`+p.localName+`(uint64(v))`)
  696. case descriptor.FieldDescriptorProto_TYPE_BOOL:
  697. sum = append(sum, strconv.Itoa(valueKeySize))
  698. sum = append(sum, `1`)
  699. case descriptor.FieldDescriptorProto_TYPE_STRING:
  700. sum = append(sum, strconv.Itoa(valueKeySize))
  701. sum = append(sum, `len(v)+sov`+p.localName+`(uint64(len(v)))`)
  702. case descriptor.FieldDescriptorProto_TYPE_BYTES:
  703. if gogoproto.IsCustomType(field) {
  704. p.P(`cSize := 0`)
  705. if gogoproto.IsNullable(field) {
  706. p.P(`if `, accessor, ` != nil {`)
  707. p.In()
  708. }
  709. p.P(`cSize = `, accessor, `.Size()`)
  710. p.P(`cSize += `, strconv.Itoa(valueKeySize), ` + sov`+p.localName+`(uint64(cSize))`)
  711. if gogoproto.IsNullable(field) {
  712. p.Out()
  713. p.P(`}`)
  714. }
  715. sum = append(sum, `cSize`)
  716. } else {
  717. p.P(`byteSize := 0`)
  718. if proto3 {
  719. p.P(`if len(v) > 0 {`)
  720. } else {
  721. p.P(`if v != nil {`)
  722. }
  723. p.In()
  724. p.P(`byteSize = `, strconv.Itoa(valueKeySize), ` + len(v)+sov`+p.localName+`(uint64(len(v)))`)
  725. p.Out()
  726. p.P(`}`)
  727. sum = append(sum, `byteSize`)
  728. }
  729. case descriptor.FieldDescriptorProto_TYPE_SINT32,
  730. descriptor.FieldDescriptorProto_TYPE_SINT64:
  731. sum = append(sum, strconv.Itoa(valueKeySize))
  732. sum = append(sum, `soz`+p.localName+`(uint64(v))`)
  733. case descriptor.FieldDescriptorProto_TYPE_MESSAGE:
  734. if valuegoTyp != valuegoAliasTyp &&
  735. !gogoproto.IsStdTime(field) &&
  736. !gogoproto.IsStdDuration(field) {
  737. if nullable {
  738. // cast back to the type that has the generated methods on it
  739. accessor = `((` + valuegoTyp + `)(` + accessor + `))`
  740. } else {
  741. accessor = `((*` + valuegoTyp + `)(&` + accessor + `))`
  742. }
  743. } else if !nullable {
  744. accessor = `(&v)`
  745. }
  746. p.P(`msgSize := 0`)
  747. p.P(`if `, accessor, ` != nil {`)
  748. p.In()
  749. if gogoproto.IsStdTime(field) {
  750. p.P(`msgSize = `, p.typesPkg.Use(), `.SizeOfStdTime(*`, accessor, `)`)
  751. } else if gogoproto.IsStdDuration(field) {
  752. p.P(`msgSize = `, p.typesPkg.Use(), `.SizeOfStdDuration(*`, accessor, `)`)
  753. } else if protoSizer {
  754. p.P(`msgSize = `, accessor, `.ProtoSize()`)
  755. } else {
  756. p.P(`msgSize = `, accessor, `.Size()`)
  757. }
  758. p.P(`msgSize += `, strconv.Itoa(valueKeySize), ` + sov`+p.localName+`(uint64(msgSize))`)
  759. p.Out()
  760. p.P(`}`)
  761. sum = append(sum, `msgSize`)
  762. }
  763. p.P(`mapSize := `, strings.Join(sum, " + "))
  764. p.callVarint("mapSize")
  765. p.encodeKey(1, wireToType(keywire))
  766. p.mapField(numGen, field, m.KeyField, "k", protoSizer)
  767. nullableMsg := nullable && (m.ValueField.GetType() == descriptor.FieldDescriptorProto_TYPE_MESSAGE ||
  768. gogoproto.IsCustomType(field) && m.ValueField.IsBytes())
  769. plainBytes := m.ValueField.IsBytes() && !gogoproto.IsCustomType(field)
  770. if nullableMsg {
  771. p.P(`if `, accessor, ` != nil { `)
  772. p.In()
  773. } else if plainBytes {
  774. if proto3 {
  775. p.P(`if len(`, accessor, `) > 0 {`)
  776. } else {
  777. p.P(`if `, accessor, ` != nil {`)
  778. }
  779. p.In()
  780. }
  781. p.encodeKey(2, wireToType(valuewire))
  782. p.mapField(numGen, field, m.ValueField, accessor, protoSizer)
  783. if nullableMsg || plainBytes {
  784. p.Out()
  785. p.P(`}`)
  786. }
  787. p.Out()
  788. p.P(`}`)
  789. } else if repeated {
  790. p.P(`for _, msg := range m.`, fieldname, ` {`)
  791. p.In()
  792. p.encodeKey(fieldNumber, wireType)
  793. varName := "msg"
  794. if gogoproto.IsStdTime(field) {
  795. if gogoproto.IsNullable(field) {
  796. varName = "*" + varName
  797. }
  798. p.callVarint(p.typesPkg.Use(), `.SizeOfStdTime(`, varName, `)`)
  799. p.P(`n, err := `, p.typesPkg.Use(), `.StdTimeMarshalTo(`, varName, `, dAtA[i:])`)
  800. } else if gogoproto.IsStdDuration(field) {
  801. if gogoproto.IsNullable(field) {
  802. varName = "*" + varName
  803. }
  804. p.callVarint(p.typesPkg.Use(), `.SizeOfStdDuration(`, varName, `)`)
  805. p.P(`n, err := `, p.typesPkg.Use(), `.StdDurationMarshalTo(`, varName, `, dAtA[i:])`)
  806. } else if protoSizer {
  807. p.callVarint(varName, ".ProtoSize()")
  808. p.P(`n, err := `, varName, `.MarshalTo(dAtA[i:])`)
  809. } else {
  810. p.callVarint(varName, ".Size()")
  811. p.P(`n, err := `, varName, `.MarshalTo(dAtA[i:])`)
  812. }
  813. p.P(`if err != nil {`)
  814. p.In()
  815. p.P(`return 0, err`)
  816. p.Out()
  817. p.P(`}`)
  818. p.P(`i+=n`)
  819. p.Out()
  820. p.P(`}`)
  821. } else {
  822. p.encodeKey(fieldNumber, wireType)
  823. varName := `m.` + fieldname
  824. if gogoproto.IsStdTime(field) {
  825. if gogoproto.IsNullable(field) {
  826. varName = "*" + varName
  827. }
  828. p.callVarint(p.typesPkg.Use(), `.SizeOfStdTime(`, varName, `)`)
  829. p.P(`n`, numGen.Next(), `, err := `, p.typesPkg.Use(), `.StdTimeMarshalTo(`, varName, `, dAtA[i:])`)
  830. } else if gogoproto.IsStdDuration(field) {
  831. if gogoproto.IsNullable(field) {
  832. varName = "*" + varName
  833. }
  834. p.callVarint(p.typesPkg.Use(), `.SizeOfStdDuration(`, varName, `)`)
  835. p.P(`n`, numGen.Next(), `, err := `, p.typesPkg.Use(), `.StdDurationMarshalTo(`, varName, `, dAtA[i:])`)
  836. } else if protoSizer {
  837. p.callVarint(varName, `.ProtoSize()`)
  838. p.P(`n`, numGen.Next(), `, err := `, varName, `.MarshalTo(dAtA[i:])`)
  839. } else {
  840. p.callVarint(varName, `.Size()`)
  841. p.P(`n`, numGen.Next(), `, err := `, varName, `.MarshalTo(dAtA[i:])`)
  842. }
  843. p.P(`if err != nil {`)
  844. p.In()
  845. p.P(`return 0, err`)
  846. p.Out()
  847. p.P(`}`)
  848. p.P(`i+=n`, numGen.Current())
  849. }
  850. case descriptor.FieldDescriptorProto_TYPE_BYTES:
  851. if !gogoproto.IsCustomType(field) {
  852. if repeated {
  853. p.P(`for _, b := range m.`, fieldname, ` {`)
  854. p.In()
  855. p.encodeKey(fieldNumber, wireType)
  856. p.callVarint("len(b)")
  857. p.P(`i+=copy(dAtA[i:], b)`)
  858. p.Out()
  859. p.P(`}`)
  860. } else if proto3 {
  861. p.P(`if len(m.`, fieldname, `) > 0 {`)
  862. p.In()
  863. p.encodeKey(fieldNumber, wireType)
  864. p.callVarint(`len(m.`, fieldname, `)`)
  865. p.P(`i+=copy(dAtA[i:], m.`, fieldname, `)`)
  866. p.Out()
  867. p.P(`}`)
  868. } else {
  869. p.encodeKey(fieldNumber, wireType)
  870. p.callVarint(`len(m.`, fieldname, `)`)
  871. p.P(`i+=copy(dAtA[i:], m.`, fieldname, `)`)
  872. }
  873. } else {
  874. if repeated {
  875. p.P(`for _, msg := range m.`, fieldname, ` {`)
  876. p.In()
  877. p.encodeKey(fieldNumber, wireType)
  878. if protoSizer {
  879. p.callVarint(`msg.ProtoSize()`)
  880. } else {
  881. p.callVarint(`msg.Size()`)
  882. }
  883. p.P(`n, err := msg.MarshalTo(dAtA[i:])`)
  884. p.P(`if err != nil {`)
  885. p.In()
  886. p.P(`return 0, err`)
  887. p.Out()
  888. p.P(`}`)
  889. p.P(`i+=n`)
  890. p.Out()
  891. p.P(`}`)
  892. } else {
  893. p.encodeKey(fieldNumber, wireType)
  894. if protoSizer {
  895. p.callVarint(`m.`, fieldname, `.ProtoSize()`)
  896. } else {
  897. p.callVarint(`m.`, fieldname, `.Size()`)
  898. }
  899. p.P(`n`, numGen.Next(), `, err := m.`, fieldname, `.MarshalTo(dAtA[i:])`)
  900. p.P(`if err != nil {`)
  901. p.In()
  902. p.P(`return 0, err`)
  903. p.Out()
  904. p.P(`}`)
  905. p.P(`i+=n`, numGen.Current())
  906. }
  907. }
  908. case descriptor.FieldDescriptorProto_TYPE_SINT32:
  909. if packed {
  910. datavar := "dAtA" + numGen.Next()
  911. jvar := "j" + numGen.Next()
  912. p.P(datavar, ` := make([]byte, len(m.`, fieldname, ")*5)")
  913. p.P(`var `, jvar, ` int`)
  914. p.P(`for _, num := range m.`, fieldname, ` {`)
  915. p.In()
  916. xvar := "x" + numGen.Next()
  917. p.P(xvar, ` := (uint32(num) << 1) ^ uint32((num >> 31))`)
  918. p.P(`for `, xvar, ` >= 1<<7 {`)
  919. p.In()
  920. p.P(datavar, `[`, jvar, `] = uint8(uint64(`, xvar, `)&0x7f|0x80)`)
  921. p.P(jvar, `++`)
  922. p.P(xvar, ` >>= 7`)
  923. p.Out()
  924. p.P(`}`)
  925. p.P(datavar, `[`, jvar, `] = uint8(`, xvar, `)`)
  926. p.P(jvar, `++`)
  927. p.Out()
  928. p.P(`}`)
  929. p.encodeKey(fieldNumber, wireType)
  930. p.callVarint(jvar)
  931. p.P(`i+=copy(dAtA[i:], `, datavar, `[:`, jvar, `])`)
  932. } else if repeated {
  933. p.P(`for _, num := range m.`, fieldname, ` {`)
  934. p.In()
  935. p.encodeKey(fieldNumber, wireType)
  936. p.P(`x`, numGen.Next(), ` := (uint32(num) << 1) ^ uint32((num >> 31))`)
  937. p.encodeVarint("x" + numGen.Current())
  938. p.Out()
  939. p.P(`}`)
  940. } else if proto3 {
  941. p.P(`if m.`, fieldname, ` != 0 {`)
  942. p.In()
  943. p.encodeKey(fieldNumber, wireType)
  944. p.callVarint(`(uint32(m.`, fieldname, `) << 1) ^ uint32((m.`, fieldname, ` >> 31))`)
  945. p.Out()
  946. p.P(`}`)
  947. } else if !nullable {
  948. p.encodeKey(fieldNumber, wireType)
  949. p.callVarint(`(uint32(m.`, fieldname, `) << 1) ^ uint32((m.`, fieldname, ` >> 31))`)
  950. } else {
  951. p.encodeKey(fieldNumber, wireType)
  952. p.callVarint(`(uint32(*m.`, fieldname, `) << 1) ^ uint32((*m.`, fieldname, ` >> 31))`)
  953. }
  954. case descriptor.FieldDescriptorProto_TYPE_SINT64:
  955. if packed {
  956. jvar := "j" + numGen.Next()
  957. xvar := "x" + numGen.Next()
  958. datavar := "dAtA" + numGen.Next()
  959. p.P(`var `, jvar, ` int`)
  960. p.P(datavar, ` := make([]byte, len(m.`, fieldname, `)*10)`)
  961. p.P(`for _, num := range m.`, fieldname, ` {`)
  962. p.In()
  963. p.P(xvar, ` := (uint64(num) << 1) ^ uint64((num >> 63))`)
  964. p.P(`for `, xvar, ` >= 1<<7 {`)
  965. p.In()
  966. p.P(datavar, `[`, jvar, `] = uint8(uint64(`, xvar, `)&0x7f|0x80)`)
  967. p.P(jvar, `++`)
  968. p.P(xvar, ` >>= 7`)
  969. p.Out()
  970. p.P(`}`)
  971. p.P(datavar, `[`, jvar, `] = uint8(`, xvar, `)`)
  972. p.P(jvar, `++`)
  973. p.Out()
  974. p.P(`}`)
  975. p.encodeKey(fieldNumber, wireType)
  976. p.callVarint(jvar)
  977. p.P(`i+=copy(dAtA[i:], `, datavar, `[:`, jvar, `])`)
  978. } else if repeated {
  979. p.P(`for _, num := range m.`, fieldname, ` {`)
  980. p.In()
  981. p.encodeKey(fieldNumber, wireType)
  982. p.P(`x`, numGen.Next(), ` := (uint64(num) << 1) ^ uint64((num >> 63))`)
  983. p.encodeVarint("x" + numGen.Current())
  984. p.Out()
  985. p.P(`}`)
  986. } else if proto3 {
  987. p.P(`if m.`, fieldname, ` != 0 {`)
  988. p.In()
  989. p.encodeKey(fieldNumber, wireType)
  990. p.callVarint(`(uint64(m.`, fieldname, `) << 1) ^ uint64((m.`, fieldname, ` >> 63))`)
  991. p.Out()
  992. p.P(`}`)
  993. } else if !nullable {
  994. p.encodeKey(fieldNumber, wireType)
  995. p.callVarint(`(uint64(m.`, fieldname, `) << 1) ^ uint64((m.`, fieldname, ` >> 63))`)
  996. } else {
  997. p.encodeKey(fieldNumber, wireType)
  998. p.callVarint(`(uint64(*m.`, fieldname, `) << 1) ^ uint64((*m.`, fieldname, ` >> 63))`)
  999. }
  1000. default:
  1001. panic("not implemented")
  1002. }
  1003. if (required && nullable) || repeated || doNilCheck {
  1004. p.Out()
  1005. p.P(`}`)
  1006. }
  1007. }
  1008. func (p *marshalto) Generate(file *generator.FileDescriptor) {
  1009. numGen := NewNumGen()
  1010. p.PluginImports = generator.NewPluginImports(p.Generator)
  1011. p.atleastOne = false
  1012. p.localName = generator.FileName(file)
  1013. p.mathPkg = p.NewImport("math")
  1014. p.sortKeysPkg = p.NewImport("github.com/gogo/protobuf/sortkeys")
  1015. p.protoPkg = p.NewImport("github.com/gogo/protobuf/proto")
  1016. if !gogoproto.ImportsGoGoProto(file.FileDescriptorProto) {
  1017. p.protoPkg = p.NewImport("github.com/golang/protobuf/proto")
  1018. }
  1019. p.errorsPkg = p.NewImport("errors")
  1020. p.binaryPkg = p.NewImport("encoding/binary")
  1021. p.typesPkg = p.NewImport("github.com/gogo/protobuf/types")
  1022. for _, message := range file.Messages() {
  1023. if message.DescriptorProto.GetOptions().GetMapEntry() {
  1024. continue
  1025. }
  1026. ccTypeName := generator.CamelCaseSlice(message.TypeName())
  1027. if !gogoproto.IsMarshaler(file.FileDescriptorProto, message.DescriptorProto) &&
  1028. !gogoproto.IsUnsafeMarshaler(file.FileDescriptorProto, message.DescriptorProto) {
  1029. continue
  1030. }
  1031. p.atleastOne = true
  1032. p.P(`func (m *`, ccTypeName, `) Marshal() (dAtA []byte, err error) {`)
  1033. p.In()
  1034. if gogoproto.IsProtoSizer(file.FileDescriptorProto, message.DescriptorProto) {
  1035. p.P(`size := m.ProtoSize()`)
  1036. } else {
  1037. p.P(`size := m.Size()`)
  1038. }
  1039. p.P(`dAtA = make([]byte, size)`)
  1040. p.P(`n, err := m.MarshalTo(dAtA)`)
  1041. p.P(`if err != nil {`)
  1042. p.In()
  1043. p.P(`return nil, err`)
  1044. p.Out()
  1045. p.P(`}`)
  1046. p.P(`return dAtA[:n], nil`)
  1047. p.Out()
  1048. p.P(`}`)
  1049. p.P(``)
  1050. p.P(`func (m *`, ccTypeName, `) MarshalTo(dAtA []byte) (int, error) {`)
  1051. p.In()
  1052. p.P(`var i int`)
  1053. p.P(`_ = i`)
  1054. p.P(`var l int`)
  1055. p.P(`_ = l`)
  1056. fields := orderFields(message.GetField())
  1057. sort.Sort(fields)
  1058. oneofs := make(map[string]struct{})
  1059. for _, field := range message.Field {
  1060. oneof := field.OneofIndex != nil
  1061. if !oneof {
  1062. proto3 := gogoproto.IsProto3(file.FileDescriptorProto)
  1063. p.generateField(proto3, numGen, file, message, field)
  1064. } else {
  1065. fieldname := p.GetFieldName(message, field)
  1066. if _, ok := oneofs[fieldname]; !ok {
  1067. oneofs[fieldname] = struct{}{}
  1068. p.P(`if m.`, fieldname, ` != nil {`)
  1069. p.In()
  1070. p.P(`nn`, numGen.Next(), `, err := m.`, fieldname, `.MarshalTo(dAtA[i:])`)
  1071. p.P(`if err != nil {`)
  1072. p.In()
  1073. p.P(`return 0, err`)
  1074. p.Out()
  1075. p.P(`}`)
  1076. p.P(`i+=nn`, numGen.Current())
  1077. p.Out()
  1078. p.P(`}`)
  1079. }
  1080. }
  1081. }
  1082. if message.DescriptorProto.HasExtension() {
  1083. if gogoproto.HasExtensionsMap(file.FileDescriptorProto, message.DescriptorProto) {
  1084. p.P(`n, err := `, p.protoPkg.Use(), `.EncodeInternalExtension(m, dAtA[i:])`)
  1085. p.P(`if err != nil {`)
  1086. p.In()
  1087. p.P(`return 0, err`)
  1088. p.Out()
  1089. p.P(`}`)
  1090. p.P(`i+=n`)
  1091. } else {
  1092. p.P(`if m.XXX_extensions != nil {`)
  1093. p.In()
  1094. p.P(`i+=copy(dAtA[i:], m.XXX_extensions)`)
  1095. p.Out()
  1096. p.P(`}`)
  1097. }
  1098. }
  1099. if gogoproto.HasUnrecognized(file.FileDescriptorProto, message.DescriptorProto) {
  1100. p.P(`if m.XXX_unrecognized != nil {`)
  1101. p.In()
  1102. p.P(`i+=copy(dAtA[i:], m.XXX_unrecognized)`)
  1103. p.Out()
  1104. p.P(`}`)
  1105. }
  1106. p.P(`return i, nil`)
  1107. p.Out()
  1108. p.P(`}`)
  1109. p.P()
  1110. //Generate MarshalTo methods for oneof fields
  1111. m := proto.Clone(message.DescriptorProto).(*descriptor.DescriptorProto)
  1112. for _, field := range m.Field {
  1113. oneof := field.OneofIndex != nil
  1114. if !oneof {
  1115. continue
  1116. }
  1117. ccTypeName := p.OneOfTypeName(message, field)
  1118. p.P(`func (m *`, ccTypeName, `) MarshalTo(dAtA []byte) (int, error) {`)
  1119. p.In()
  1120. p.P(`i := 0`)
  1121. vanity.TurnOffNullableForNativeTypes(field)
  1122. p.generateField(false, numGen, file, message, field)
  1123. p.P(`return i, nil`)
  1124. p.Out()
  1125. p.P(`}`)
  1126. }
  1127. }
  1128. if p.atleastOne {
  1129. p.P(`func encodeVarint`, p.localName, `(dAtA []byte, offset int, v uint64) int {`)
  1130. p.In()
  1131. p.P(`for v >= 1<<7 {`)
  1132. p.In()
  1133. p.P(`dAtA[offset] = uint8(v&0x7f|0x80)`)
  1134. p.P(`v >>= 7`)
  1135. p.P(`offset++`)
  1136. p.Out()
  1137. p.P(`}`)
  1138. p.P(`dAtA[offset] = uint8(v)`)
  1139. p.P(`return offset+1`)
  1140. p.Out()
  1141. p.P(`}`)
  1142. }
  1143. }
  1144. func init() {
  1145. generator.RegisterPlugin(NewMarshal())
  1146. }