| 
									
										
										
										
											2015-02-13 14:40:36 -05:00
										 |  |  | // Copyright 2009 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. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | package gc | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | //	case OADD: | 
					
						
							|  |  |  | //		if(n->right->op == OLITERAL) { | 
					
						
							|  |  |  | //			v = n->right->vconst; | 
					
						
							|  |  |  | //			naddr(n->left, a, canemitcode); | 
					
						
							|  |  |  | //		} else | 
					
						
							|  |  |  | //		if(n->left->op == OLITERAL) { | 
					
						
							|  |  |  | //			v = n->left->vconst; | 
					
						
							|  |  |  | //			naddr(n->right, a, canemitcode); | 
					
						
							|  |  |  | //		} else | 
					
						
							|  |  |  | //			goto bad; | 
					
						
							|  |  |  | //		a->offset += v; | 
					
						
							|  |  |  | //		break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-10-22 09:51:12 +09:00
										 |  |  | // a function named init is a special case. | 
					
						
							|  |  |  | // it is called by the initialization before | 
					
						
							|  |  |  | // main is run. to make it unique within a | 
					
						
							|  |  |  | // package and also uncallable, the name, | 
					
						
							|  |  |  | // normally "pkg.init", is altered to "pkg.init.1". | 
					
						
							| 
									
										
										
										
											2015-02-13 14:40:36 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | var renameinit_initgen int | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func renameinit() *Sym { | 
					
						
							|  |  |  | 	renameinit_initgen++ | 
					
						
							| 
									
										
										
										
											2015-03-06 12:02:24 -08:00
										 |  |  | 	return Lookupf("init.%d", renameinit_initgen) | 
					
						
							| 
									
										
										
										
											2015-02-13 14:40:36 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-10-22 09:51:12 +09:00
										 |  |  | // hand-craft the following initialization code | 
					
						
							|  |  |  | //	var initdoneĀ· uint8 				(1) | 
					
						
							|  |  |  | //	func init()					(2) | 
					
						
							| 
									
										
										
										
											2016-01-26 15:55:05 -08:00
										 |  |  | //              if initdoneĀ· > 1 {                      (3) | 
					
						
							|  |  |  | //                      return                          (3a) | 
					
						
							|  |  |  | //		if initdoneĀ· == 1 {			(4) | 
					
						
							|  |  |  | //			throw();			(4a) | 
					
						
							| 
									
										
										
										
											2015-10-22 09:51:12 +09:00
										 |  |  | //		} | 
					
						
							|  |  |  | //		initdoneĀ· = 1;				(6) | 
					
						
							|  |  |  | //		// over all matching imported symbols | 
					
						
							|  |  |  | //			<pkg>.init()			(7) | 
					
						
							|  |  |  | //		{ <init stmts> }			(8) | 
					
						
							|  |  |  | //		init.<n>() // if any			(9) | 
					
						
							|  |  |  | //		initdoneĀ· = 2;				(10) | 
					
						
							|  |  |  | //		return					(11) | 
					
						
							|  |  |  | //	} | 
					
						
							| 
									
										
										
										
											2016-02-27 14:31:33 -08:00
										 |  |  | func anyinit(n []*Node) bool { | 
					
						
							| 
									
										
										
										
											2015-02-13 14:40:36 -05:00
										 |  |  | 	// are there any interesting init statements | 
					
						
							| 
									
										
										
										
											2016-02-27 14:31:33 -08:00
										 |  |  | 	for _, ln := range n { | 
					
						
							|  |  |  | 		switch ln.Op { | 
					
						
							| 
									
										
										
										
											2015-04-01 09:38:44 -07:00
										 |  |  | 		case ODCLFUNC, ODCLCONST, ODCLTYPE, OEMPTY: | 
					
						
							| 
									
										
										
										
											2015-02-13 14:40:36 -05:00
										 |  |  | 			break | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												cmd/internal/gc: emit write barriers at lower level
This is primarily preparation for inlining, not an optimization by itself,
but it still helps some.
name                                       old                     new          delta
BenchmarkBinaryTree17              18.2s Ć (0.99,1.01)     17.9s Ć (0.99,1.01)  -1.57%
BenchmarkFannkuch11                4.44s Ć (1.00,1.00)     4.42s Ć (1.00,1.00)  -0.40%
BenchmarkFmtFprintfEmpty           119ns Ć (0.95,1.02)     118ns Ć (0.96,1.02)  ~
BenchmarkFmtFprintfString          501ns Ć (0.99,1.02)     486ns Ć (0.99,1.01)  -2.89%
BenchmarkFmtFprintfInt             474ns Ć (0.99,1.00)     457ns Ć (0.99,1.01)  -3.59%
BenchmarkFmtFprintfIntInt          792ns Ć (1.00,1.00)     768ns Ć (1.00,1.01)  -3.03%
BenchmarkFmtFprintfPrefixedInt     574ns Ć (1.00,1.01)     584ns Ć (0.99,1.03)  +1.83%
BenchmarkFmtFprintfFloat           749ns Ć (1.00,1.00)     739ns Ć (0.99,1.00)  -1.34%
BenchmarkFmtManyArgs              2.94µs à (1.00,1.01)    2.77µs à (1.00,1.00)  -5.76%
BenchmarkGobDecode                39.5ms Ć (0.99,1.01)    39.3ms Ć (0.99,1.01)  ~
BenchmarkGobEncode                39.4ms Ć (1.00,1.01)    39.4ms Ć (0.99,1.00)  ~
BenchmarkGzip                      658ms Ć (1.00,1.01)     661ms Ć (0.99,1.01)  ~
BenchmarkGunzip                    142ms Ć (1.00,1.00)     142ms Ć (1.00,1.00)  +0.22%
BenchmarkHTTPClientServer          134µs à (0.99,1.01)     133µs à (0.98,1.01)  ~
BenchmarkJSONEncode               57.1ms Ć (0.99,1.01)    56.5ms Ć (0.99,1.01)  ~
BenchmarkJSONDecode                141ms Ć (1.00,1.00)     143ms Ć (1.00,1.00)  +1.09%
BenchmarkMandelbrot200            6.01ms Ć (1.00,1.00)    6.01ms Ć (1.00,1.00)  ~
BenchmarkGoParse                  10.1ms Ć (0.91,1.09)     9.6ms Ć (0.94,1.07)  ~
BenchmarkRegexpMatchEasy0_32       207ns Ć (1.00,1.01)     210ns Ć (1.00,1.00)  +1.45%
BenchmarkRegexpMatchEasy0_1K       592ns Ć (0.99,1.00)     596ns Ć (0.99,1.01)  +0.68%
BenchmarkRegexpMatchEasy1_32       184ns Ć (0.99,1.01)     184ns Ć (0.99,1.01)  ~
BenchmarkRegexpMatchEasy1_1K      1.01µs à (1.00,1.00)    1.01µs à (0.99,1.01)  ~
BenchmarkRegexpMatchMedium_32      327ns Ć (0.99,1.00)     327ns Ć (1.00,1.01)  ~
BenchmarkRegexpMatchMedium_1K     92.5µs à (1.00,1.00)    93.0µs à (1.00,1.02)  +0.48%
BenchmarkRegexpMatchHard_32       4.79µs à (0.95,1.00)    4.76µs à (0.95,1.01)  ~
BenchmarkRegexpMatchHard_1K        136µs à (1.00,1.00)     136µs à (1.00,1.01)  ~
BenchmarkRevcomp                   900ms Ć (0.99,1.01)     892ms Ć (1.00,1.01)  ~
BenchmarkTemplate                  170ms Ć (0.99,1.01)     175ms Ć (0.99,1.00)  +2.95%
BenchmarkTimeParse                 645ns Ć (1.00,1.00)     638ns Ć (1.00,1.00)  -1.16%
BenchmarkTimeFormat                740ns Ć (1.00,1.00)     772ns Ć (1.00,1.00)  +4.39%
Change-Id: I0be905e32791e0cb70ff01f169c4b309a971d981
Reviewed-on: https://go-review.googlesource.com/9159
Reviewed-by: Rick Hudson <rlh@golang.org>
Run-TryBot: Russ Cox <rsc@golang.org>
TryBot-Result: Gobot Gobot <gobot@golang.org>
											
										 
											2015-04-17 00:25:10 -04:00
										 |  |  | 		case OAS, OASWB: | 
					
						
							| 
									
										
										
										
											2016-02-27 14:31:33 -08:00
										 |  |  | 			if isblank(ln.Left) && candiscard(ln.Right) { | 
					
						
							| 
									
										
										
										
											2015-02-13 14:40:36 -05:00
										 |  |  | 				break | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			fallthrough | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			// fall through | 
					
						
							|  |  |  | 		default: | 
					
						
							| 
									
										
										
										
											2015-02-17 22:13:49 -05:00
										 |  |  | 			return true | 
					
						
							| 
									
										
										
										
											2015-02-13 14:40:36 -05:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// is this main | 
					
						
							|  |  |  | 	if localpkg.Name == "main" { | 
					
						
							| 
									
										
										
										
											2015-02-17 22:13:49 -05:00
										 |  |  | 		return true | 
					
						
							| 
									
										
										
										
											2015-02-13 14:40:36 -05:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// is there an explicit init function | 
					
						
							| 
									
										
										
										
											2015-02-23 16:07:24 -05:00
										 |  |  | 	s := Lookup("init.1") | 
					
						
							| 
									
										
										
										
											2015-02-13 14:40:36 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if s.Def != nil { | 
					
						
							| 
									
										
										
										
											2015-02-17 22:13:49 -05:00
										 |  |  | 		return true | 
					
						
							| 
									
										
										
										
											2015-02-13 14:40:36 -05:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// are there any imported init functions | 
					
						
							| 
									
										
										
										
											2015-03-02 16:21:15 -05:00
										 |  |  | 	for _, s := range initSyms { | 
					
						
							|  |  |  | 		if s.Def != nil { | 
					
						
							| 
									
										
										
										
											2015-02-17 22:13:49 -05:00
										 |  |  | 			return true | 
					
						
							| 
									
										
										
										
											2015-02-13 14:40:36 -05:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// then none | 
					
						
							| 
									
										
										
										
											2015-02-17 22:13:49 -05:00
										 |  |  | 	return false | 
					
						
							| 
									
										
										
										
											2015-02-13 14:40:36 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-09 20:29:21 -08:00
										 |  |  | func fninit(n []*Node) { | 
					
						
							| 
									
										
										
										
											2015-02-13 14:40:36 -05:00
										 |  |  | 	if Debug['A'] != 0 { | 
					
						
							|  |  |  | 		// sys.go or unsafe.go during compiler build | 
					
						
							|  |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-09 20:29:21 -08:00
										 |  |  | 	nf := initfix(n) | 
					
						
							| 
									
										
										
										
											2016-02-27 14:31:33 -08:00
										 |  |  | 	if !anyinit(nf) { | 
					
						
							| 
									
										
										
										
											2015-02-13 14:40:36 -05:00
										 |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-27 14:31:33 -08:00
										 |  |  | 	var r []*Node | 
					
						
							| 
									
										
										
										
											2015-02-13 14:40:36 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// (1) | 
					
						
							| 
									
										
										
										
											2015-03-06 12:02:24 -08:00
										 |  |  | 	gatevar := newname(Lookup("initdoneĀ·")) | 
					
						
							| 
									
										
										
										
											2015-02-13 14:40:36 -05:00
										 |  |  | 	addvar(gatevar, Types[TUINT8], PEXTERN) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// (2) | 
					
						
							|  |  |  | 	Maxarg = 0 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-23 16:07:24 -05:00
										 |  |  | 	fn := Nod(ODCLFUNC, nil, nil) | 
					
						
							| 
									
										
										
										
											2015-03-06 12:02:24 -08:00
										 |  |  | 	initsym := Lookup("init") | 
					
						
							| 
									
										
										
										
											2015-05-27 10:42:55 -04:00
										 |  |  | 	fn.Func.Nname = newname(initsym) | 
					
						
							|  |  |  | 	fn.Func.Nname.Name.Defn = fn | 
					
						
							|  |  |  | 	fn.Func.Nname.Name.Param.Ntype = Nod(OTFUNC, nil, nil) | 
					
						
							|  |  |  | 	declare(fn.Func.Nname, PFUNC) | 
					
						
							| 
									
										
										
										
											2015-02-13 14:40:36 -05:00
										 |  |  | 	funchdr(fn) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// (3) | 
					
						
							| 
									
										
										
										
											2015-02-23 16:07:24 -05:00
										 |  |  | 	a := Nod(OIF, nil, nil) | 
					
						
							| 
									
										
										
										
											2016-01-26 15:55:05 -08:00
										 |  |  | 	a.Left = Nod(OGT, gatevar, Nodintconst(1)) | 
					
						
							|  |  |  | 	a.Likely = 1 | 
					
						
							| 
									
										
										
										
											2016-02-27 14:31:33 -08:00
										 |  |  | 	r = append(r, a) | 
					
						
							| 
									
										
										
										
											2016-01-26 15:55:05 -08:00
										 |  |  | 	// (3a) | 
					
						
							| 
									
										
										
										
											2016-03-01 12:50:17 -08:00
										 |  |  | 	a.Nbody.Set([]*Node{Nod(ORETURN, nil, nil)}) | 
					
						
							| 
									
										
										
										
											2015-02-13 14:40:36 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// (4) | 
					
						
							| 
									
										
										
										
											2015-02-23 16:07:24 -05:00
										 |  |  | 	b := Nod(OIF, nil, nil) | 
					
						
							| 
									
										
										
										
											2016-01-26 15:55:05 -08:00
										 |  |  | 	b.Left = Nod(OEQ, gatevar, Nodintconst(1)) | 
					
						
							|  |  |  | 	// this actually isn't likely, but code layout is better | 
					
						
							|  |  |  | 	// like this: no JMP needed after the call. | 
					
						
							|  |  |  | 	b.Likely = 1 | 
					
						
							| 
									
										
										
										
											2016-03-01 12:50:17 -08:00
										 |  |  | 	r = append(r, b) | 
					
						
							| 
									
										
										
										
											2016-01-26 15:55:05 -08:00
										 |  |  | 	// (4a) | 
					
						
							| 
									
										
										
										
											2016-03-04 15:19:06 -08:00
										 |  |  | 	b.Nbody.Set([]*Node{Nod(OCALL, syslook("throwinit"), nil)}) | 
					
						
							| 
									
										
										
										
											2015-02-13 14:40:36 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// (6) | 
					
						
							|  |  |  | 	a = Nod(OAS, gatevar, Nodintconst(1)) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-27 14:31:33 -08:00
										 |  |  | 	r = append(r, a) | 
					
						
							| 
									
										
										
										
											2015-02-13 14:40:36 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// (7) | 
					
						
							| 
									
										
										
										
											2015-03-02 16:21:15 -05:00
										 |  |  | 	for _, s := range initSyms { | 
					
						
							|  |  |  | 		if s.Def != nil && s != initsym { | 
					
						
							| 
									
										
										
										
											2015-02-13 14:40:36 -05:00
										 |  |  | 			// could check that it is fn of no args/returns | 
					
						
							|  |  |  | 			a = Nod(OCALL, s.Def, nil) | 
					
						
							| 
									
										
										
										
											2016-02-27 14:31:33 -08:00
										 |  |  | 			r = append(r, a) | 
					
						
							| 
									
										
										
										
											2015-02-13 14:40:36 -05:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// (8) | 
					
						
							| 
									
										
										
										
											2016-02-27 14:31:33 -08:00
										 |  |  | 	r = append(r, nf...) | 
					
						
							| 
									
										
										
										
											2015-02-13 14:40:36 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// (9) | 
					
						
							|  |  |  | 	// could check that it is fn of no args/returns | 
					
						
							| 
									
										
										
										
											2015-02-23 16:07:24 -05:00
										 |  |  | 	for i := 1; ; i++ { | 
					
						
							| 
									
										
										
										
											2015-03-06 12:02:24 -08:00
										 |  |  | 		s := Lookupf("init.%d", i) | 
					
						
							| 
									
										
										
										
											2015-02-13 14:40:36 -05:00
										 |  |  | 		if s.Def == nil { | 
					
						
							|  |  |  | 			break | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		a = Nod(OCALL, s.Def, nil) | 
					
						
							| 
									
										
										
										
											2016-02-27 14:31:33 -08:00
										 |  |  | 		r = append(r, a) | 
					
						
							| 
									
										
										
										
											2015-02-13 14:40:36 -05:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// (10) | 
					
						
							|  |  |  | 	a = Nod(OAS, gatevar, Nodintconst(2)) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-27 14:31:33 -08:00
										 |  |  | 	r = append(r, a) | 
					
						
							| 
									
										
										
										
											2015-02-13 14:40:36 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// (11) | 
					
						
							|  |  |  | 	a = Nod(ORETURN, nil, nil) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-27 14:31:33 -08:00
										 |  |  | 	r = append(r, a) | 
					
						
							| 
									
										
										
										
											2015-05-27 10:42:55 -04:00
										 |  |  | 	exportsym(fn.Func.Nname) | 
					
						
							| 
									
										
										
										
											2015-02-13 14:40:36 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-27 14:31:33 -08:00
										 |  |  | 	fn.Nbody.Set(r) | 
					
						
							| 
									
										
										
										
											2015-02-13 14:40:36 -05:00
										 |  |  | 	funcbody(fn) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	Curfn = fn | 
					
						
							|  |  |  | 	typecheck(&fn, Etop) | 
					
						
							| 
									
										
										
										
											2016-02-27 14:31:33 -08:00
										 |  |  | 	typecheckslice(r, Etop) | 
					
						
							| 
									
										
										
										
											2015-02-13 14:40:36 -05:00
										 |  |  | 	Curfn = nil | 
					
						
							|  |  |  | 	funccompile(fn) | 
					
						
							|  |  |  | } |