// Copyright 2018 The Go Authors. All rights reserved.
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file.

// Code generated by protoc-gen-go. DO NOT EDIT.
// source: cmd/protoc-gen-go/testdata/proto2/proto2.proto

package proto2

import (
	protoreflect "google.golang.org/protobuf/reflect/protoreflect"
	protoimpl "google.golang.org/protobuf/runtime/protoimpl"
	reflect "reflect"
	sync "sync"
	unsafe "unsafe"
)

type Message struct {
	state         protoimpl.MessageState `protogen:"open.v1"`
	I32           *int32                 `protobuf:"varint,1,opt,name=i32" json:"i32,omitempty"`
	M             *Message               `protobuf:"bytes,2,opt,name=m" json:"m,omitempty"`
	unknownFields protoimpl.UnknownFields
	sizeCache     protoimpl.SizeCache
}

func (x *Message) Reset() {
	*x = Message{}
	mi := &file_cmd_protoc_gen_go_testdata_proto2_proto2_proto_msgTypes[0]
	ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x))
	ms.StoreMessageInfo(mi)
}

func (x *Message) String() string {
	return protoimpl.X.MessageStringOf(x)
}

func (*Message) ProtoMessage() {}

func (x *Message) ProtoReflect() protoreflect.Message {
	mi := &file_cmd_protoc_gen_go_testdata_proto2_proto2_proto_msgTypes[0]
	if x != nil {
		ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x))
		if ms.LoadMessageInfo() == nil {
			ms.StoreMessageInfo(mi)
		}
		return ms
	}
	return mi.MessageOf(x)
}

// Deprecated: Use Message.ProtoReflect.Descriptor instead.
func (*Message) Descriptor() ([]byte, []int) {
	return file_cmd_protoc_gen_go_testdata_proto2_proto2_proto_rawDescGZIP(), []int{0}
}

func (x *Message) GetI32() int32 {
	if x != nil && x.I32 != nil {
		return *x.I32
	}
	return 0
}

func (x *Message) GetM() *Message {
	if x != nil {
		return x.M
	}
	return nil
}

var File_cmd_protoc_gen_go_testdata_proto2_proto2_proto protoreflect.FileDescriptor

const file_cmd_protoc_gen_go_testdata_proto2_proto2_proto_rawDesc = "" +
	"\n" +
	".cmd/protoc-gen-go/testdata/proto2/proto2.proto\x12\x15goproto.protoc.proto2\"I\n" +
	"\aMessage\x12\x10\n" +
	"\x03i32\x18\x01 \x01(\x05R\x03i32\x12,\n" +
	"\x01m\x18\x02 \x01(\v2\x1e.goproto.protoc.proto2.MessageR\x01mB>Z<google.golang.org/protobuf/cmd/protoc-gen-go/testdata/proto2"

var (
	file_cmd_protoc_gen_go_testdata_proto2_proto2_proto_rawDescOnce sync.Once
	file_cmd_protoc_gen_go_testdata_proto2_proto2_proto_rawDescData []byte
)

func file_cmd_protoc_gen_go_testdata_proto2_proto2_proto_rawDescGZIP() []byte {
	file_cmd_protoc_gen_go_testdata_proto2_proto2_proto_rawDescOnce.Do(func() {
		file_cmd_protoc_gen_go_testdata_proto2_proto2_proto_rawDescData = protoimpl.X.CompressGZIP(unsafe.Slice(unsafe.StringData(file_cmd_protoc_gen_go_testdata_proto2_proto2_proto_rawDesc), len(file_cmd_protoc_gen_go_testdata_proto2_proto2_proto_rawDesc)))
	})
	return file_cmd_protoc_gen_go_testdata_proto2_proto2_proto_rawDescData
}

var file_cmd_protoc_gen_go_testdata_proto2_proto2_proto_msgTypes = make([]protoimpl.MessageInfo, 1)
var file_cmd_protoc_gen_go_testdata_proto2_proto2_proto_goTypes = []any{
	(*Message)(nil), // 0: goproto.protoc.proto2.Message
}
var file_cmd_protoc_gen_go_testdata_proto2_proto2_proto_depIdxs = []int32{
	0, // 0: goproto.protoc.proto2.Message.m:type_name -> goproto.protoc.proto2.Message
	1, // [1:1] is the sub-list for method output_type
	1, // [1:1] is the sub-list for method input_type
	1, // [1:1] is the sub-list for extension type_name
	1, // [1:1] is the sub-list for extension extendee
	0, // [0:1] is the sub-list for field type_name
}

func init() { file_cmd_protoc_gen_go_testdata_proto2_proto2_proto_init() }
func file_cmd_protoc_gen_go_testdata_proto2_proto2_proto_init() {
	if File_cmd_protoc_gen_go_testdata_proto2_proto2_proto != nil {
		return
	}
	type x struct{}
	out := protoimpl.TypeBuilder{
		File: protoimpl.DescBuilder{
			GoPackagePath: reflect.TypeOf(x{}).PkgPath(),
			RawDescriptor: unsafe.Slice(unsafe.StringData(file_cmd_protoc_gen_go_testdata_proto2_proto2_proto_rawDesc), len(file_cmd_protoc_gen_go_testdata_proto2_proto2_proto_rawDesc)),
			NumEnums:      0,
			NumMessages:   1,
			NumExtensions: 0,
			NumServices:   0,
		},
		GoTypes:           file_cmd_protoc_gen_go_testdata_proto2_proto2_proto_goTypes,
		DependencyIndexes: file_cmd_protoc_gen_go_testdata_proto2_proto2_proto_depIdxs,
		MessageInfos:      file_cmd_protoc_gen_go_testdata_proto2_proto2_proto_msgTypes,
	}.Build()
	File_cmd_protoc_gen_go_testdata_proto2_proto2_proto = out.File
	file_cmd_protoc_gen_go_testdata_proto2_proto2_proto_goTypes = nil
	file_cmd_protoc_gen_go_testdata_proto2_proto2_proto_depIdxs = nil
}
