package ssa
 
 import (
+       "bytes"
        "cmd/internal/objabi"
        "cmd/internal/src"
        "fmt"
        "os"
        "regexp"
        "runtime"
+       "sort"
        "strings"
        "time"
 )
                }
        }
 
+       if f.ruleMatches != nil {
+               var keys []string
+               for key := range f.ruleMatches {
+                       keys = append(keys, key)
+               }
+               sort.Strings(keys)
+               buf := new(bytes.Buffer)
+               fmt.Fprintf(buf, "%s: ", f.Name)
+               for _, key := range keys {
+                       fmt.Fprintf(buf, "%s=%d ", key, f.ruleMatches[key])
+               }
+               fmt.Fprint(buf, "\n")
+               fmt.Print(buf.String())
+       }
+
        // Squash error printing defer
        phaseName = ""
 }
 
        // Given an environment variable used for debug hash match,
        // what file (if any) receives the yes/no logging?
        logfiles       map[string]writeSyncer
-       HTMLWriter     *HTMLWriter // html writer, for debugging
-       DebugTest      bool        // default true unless $GOSSAHASH != ""; as a debugging aid, make new code conditional on this and use GOSSAHASH to binary search for failing cases
-       PrintOrHtmlSSA bool        // true if GOSSAFUNC matches, true even if fe.Log() (spew phase results to stdout) is false.
+       HTMLWriter     *HTMLWriter    // html writer, for debugging
+       DebugTest      bool           // default true unless $GOSSAHASH != ""; as a debugging aid, make new code conditional on this and use GOSSAHASH to binary search for failing cases
+       PrintOrHtmlSSA bool           // true if GOSSAFUNC matches, true even if fe.Log() (spew phase results to stdout) is false.
+       ruleMatches    map[string]int // number of times countRule was called during compilation for any given string
 
        scheduled bool // Values in Blocks are in final order
        laidout   bool // Blocks are ordered
 
        return true
 }
 
+// countRule increments Func.ruleMatches[key].
+// If Func.ruleMatches is non-nil at the end
+// of compilation, it will be printed to stdout.
+// This is intended to make it easier to find which functions
+// which contain lots of rules matches when developing new rules.
+func countRule(v *Value, key string) bool {
+       f := v.Block.Func
+       if f.ruleMatches == nil {
+               f.ruleMatches = make(map[string]int)
+       }
+       f.ruleMatches[key]++
+       return true
+}
+
 // warnRule generates compiler debug output with string s when
 // v is not in autogenerated code, cond is true and the rule has fired.
 func warnRule(cond bool, v *Value, s string) bool {