// 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/import_public/b.proto

package import_public

import (
	sub "google.golang.org/protobuf/cmd/protoc-gen-go/testdata/import_public/sub"
	protoreflect "google.golang.org/protobuf/reflect/protoreflect"
	protoimpl "google.golang.org/protobuf/runtime/protoimpl"
	reflect "reflect"
	sync "sync"
	unsafe "unsafe"
)

type Local struct {
	state         protoimpl.MessageState `protogen:"open.v1"`
	M             *sub.M                 `protobuf:"bytes,1,opt,name=m" json:"m,omitempty"`
	E             *sub.E                 `protobuf:"varint,2,opt,name=e,enum=goproto.protoc.import_public.sub.E" json:"e,omitempty"`
	unknownFields protoimpl.UnknownFields
	sizeCache     protoimpl.SizeCache
}

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

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

func (*Local) ProtoMessage() {}

func (x *Local) ProtoReflect() protoreflect.Message {
	mi := &file_cmd_protoc_gen_go_testdata_import_public_b_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 Local.ProtoReflect.Descriptor instead.
func (*Local) Descriptor() ([]byte, []int) {
	return file_cmd_protoc_gen_go_testdata_import_public_b_proto_rawDescGZIP(), []int{0}
}

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

func (x *Local) GetE() sub.E {
	if x != nil && x.E != nil {
		return *x.E
	}
	return sub.E(0)
}

var File_cmd_protoc_gen_go_testdata_import_public_b_proto protoreflect.FileDescriptor

const file_cmd_protoc_gen_go_testdata_import_public_b_proto_rawDesc = "" +
	"\n" +
	"0cmd/protoc-gen-go/testdata/import_public/b.proto\x12\x1cgoproto.protoc.import_public\x1a4cmd/protoc-gen-go/testdata/import_public/sub/a.proto\"m\n" +
	"\x05Local\x121\n" +
	"\x01m\x18\x01 \x01(\v2#.goproto.protoc.import_public.sub.MR\x01m\x121\n" +
	"\x01e\x18\x02 \x01(\x0e2#.goproto.protoc.import_public.sub.ER\x01eBEZCgoogle.golang.org/protobuf/cmd/protoc-gen-go/testdata/import_public"

var (
	file_cmd_protoc_gen_go_testdata_import_public_b_proto_rawDescOnce sync.Once
	file_cmd_protoc_gen_go_testdata_import_public_b_proto_rawDescData []byte
)

func file_cmd_protoc_gen_go_testdata_import_public_b_proto_rawDescGZIP() []byte {
	file_cmd_protoc_gen_go_testdata_import_public_b_proto_rawDescOnce.Do(func() {
		file_cmd_protoc_gen_go_testdata_import_public_b_proto_rawDescData = protoimpl.X.CompressGZIP(unsafe.Slice(unsafe.StringData(file_cmd_protoc_gen_go_testdata_import_public_b_proto_rawDesc), len(file_cmd_protoc_gen_go_testdata_import_public_b_proto_rawDesc)))
	})
	return file_cmd_protoc_gen_go_testdata_import_public_b_proto_rawDescData
}

var file_cmd_protoc_gen_go_testdata_import_public_b_proto_msgTypes = make([]protoimpl.MessageInfo, 1)
var file_cmd_protoc_gen_go_testdata_import_public_b_proto_goTypes = []any{
	(*Local)(nil), // 0: goproto.protoc.import_public.Local
	(*sub.M)(nil), // 1: goproto.protoc.import_public.sub.M
	(sub.E)(0),    // 2: goproto.protoc.import_public.sub.E
}
var file_cmd_protoc_gen_go_testdata_import_public_b_proto_depIdxs = []int32{
	1, // 0: goproto.protoc.import_public.Local.m:type_name -> goproto.protoc.import_public.sub.M
	2, // 1: goproto.protoc.import_public.Local.e:type_name -> goproto.protoc.import_public.sub.E
	2, // [2:2] is the sub-list for method output_type
	2, // [2:2] is the sub-list for method input_type
	2, // [2:2] is the sub-list for extension type_name
	2, // [2:2] is the sub-list for extension extendee
	0, // [0:2] is the sub-list for field type_name
}

func init() { file_cmd_protoc_gen_go_testdata_import_public_b_proto_init() }
func file_cmd_protoc_gen_go_testdata_import_public_b_proto_init() {
	if File_cmd_protoc_gen_go_testdata_import_public_b_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_import_public_b_proto_rawDesc), len(file_cmd_protoc_gen_go_testdata_import_public_b_proto_rawDesc)),
			NumEnums:      0,
			NumMessages:   1,
			NumExtensions: 0,
			NumServices:   0,
		},
		GoTypes:           file_cmd_protoc_gen_go_testdata_import_public_b_proto_goTypes,
		DependencyIndexes: file_cmd_protoc_gen_go_testdata_import_public_b_proto_depIdxs,
		MessageInfos:      file_cmd_protoc_gen_go_testdata_import_public_b_proto_msgTypes,
	}.Build()
	File_cmd_protoc_gen_go_testdata_import_public_b_proto = out.File
	file_cmd_protoc_gen_go_testdata_import_public_b_proto_goTypes = nil
	file_cmd_protoc_gen_go_testdata_import_public_b_proto_depIdxs = nil
}
