blob: 8a0b1c96ec1332557ff89ddcd5f38896658c48ba [file] [log] [blame]
Jingwen Chen164e0862021-02-19 00:48:40 -05001package bp2build
2
3import (
Jingwen Chen164e0862021-02-19 00:48:40 -05004 "fmt"
Alex Márquez Pérez Muñíz Díaz Púras Thaureaux947fdbf2021-11-10 09:55:20 -05005 "os"
6 "path/filepath"
Liz Kammer6eff3232021-08-26 08:37:59 -04007 "strings"
Chris Parsons91b81f02021-12-08 11:19:06 -05008
9 "android/soong/android"
Alex Márquez Pérez Muñíz Díaz Púras Thaureaux947fdbf2021-11-10 09:55:20 -050010 "android/soong/shared"
11 "android/soong/ui/metrics/bp2build_metrics_proto"
Chris Parsons492bd912022-01-20 12:55:05 -050012 "github.com/google/blueprint"
Jingwen Chen164e0862021-02-19 00:48:40 -050013)
14
15// Simple metrics struct to collect information about a Blueprint to BUILD
16// conversion process.
17type CodegenMetrics struct {
Jingwen Chen310bc8f2021-09-20 10:54:27 +000018 // Total number of Soong modules converted to generated targets
Alex Márquez Pérez Muñíz Díaz Púras Thaureaux947fdbf2021-11-10 09:55:20 -050019 generatedModuleCount uint64
Jingwen Chen310bc8f2021-09-20 10:54:27 +000020
21 // Total number of Soong modules converted to handcrafted targets
Alex Márquez Pérez Muñíz Díaz Púras Thaureaux947fdbf2021-11-10 09:55:20 -050022 handCraftedModuleCount uint64
Jingwen Chen310bc8f2021-09-20 10:54:27 +000023
24 // Total number of unconverted Soong modules
Alex Márquez Pérez Muñíz Díaz Púras Thaureaux947fdbf2021-11-10 09:55:20 -050025 unconvertedModuleCount uint64
Jingwen Chen164e0862021-02-19 00:48:40 -050026
27 // Counts of generated Bazel targets per Bazel rule class
Alex Márquez Pérez Muñíz Díaz Púras Thaureaux947fdbf2021-11-10 09:55:20 -050028 ruleClassCount map[string]uint64
Liz Kammer6eff3232021-08-26 08:37:59 -040029
Alex Márquez Pérez Muñíz Díaz Púras Thaureaux947fdbf2021-11-10 09:55:20 -050030 // List of modules with unconverted deps
31 // NOTE: NOT in the .proto
Liz Kammer6eff3232021-08-26 08:37:59 -040032 moduleWithUnconvertedDepsMsgs []string
Jingwen Chen61174502021-09-17 08:40:45 +000033
Liz Kammerdaa09ef2021-12-15 15:35:38 -050034 // List of modules with missing deps
35 // NOTE: NOT in the .proto
36 moduleWithMissingDepsMsgs []string
37
Alex Márquez Pérez Muñíz Díaz Púras Thaureaux947fdbf2021-11-10 09:55:20 -050038 // List of converted modules
Jingwen Chen61174502021-09-17 08:40:45 +000039 convertedModules []string
Chris Parsons492bd912022-01-20 12:55:05 -050040
41 // Counts of converted modules by module type.
42 convertedModuleTypeCount map[string]uint64
43
44 // Counts of total modules by module type.
45 totalModuleTypeCount map[string]uint64
Jingwen Chen164e0862021-02-19 00:48:40 -050046}
47
Alex Márquez Pérez Muñíz Díaz Púras Thaureaux947fdbf2021-11-10 09:55:20 -050048// Serialize returns the protoized version of CodegenMetrics: bp2build_metrics_proto.Bp2BuildMetrics
49func (metrics *CodegenMetrics) Serialize() bp2build_metrics_proto.Bp2BuildMetrics {
50 return bp2build_metrics_proto.Bp2BuildMetrics{
Chris Parsons492bd912022-01-20 12:55:05 -050051 GeneratedModuleCount: metrics.generatedModuleCount,
52 HandCraftedModuleCount: metrics.handCraftedModuleCount,
53 UnconvertedModuleCount: metrics.unconvertedModuleCount,
54 RuleClassCount: metrics.ruleClassCount,
55 ConvertedModules: metrics.convertedModules,
56 ConvertedModuleTypeCount: metrics.convertedModuleTypeCount,
57 TotalModuleTypeCount: metrics.totalModuleTypeCount,
Alex Márquez Pérez Muñíz Díaz Púras Thaureaux947fdbf2021-11-10 09:55:20 -050058 }
59}
60
Jingwen Chen164e0862021-02-19 00:48:40 -050061// Print the codegen metrics to stdout.
Jingwen Chenafb84bd2021-09-20 10:31:46 +000062func (metrics *CodegenMetrics) Print() {
Alex Márquez Pérez Muñíz Díaz Púras Thaureaux947fdbf2021-11-10 09:55:20 -050063 generatedTargetCount := uint64(0)
Jingwen Chen310bc8f2021-09-20 10:54:27 +000064 for _, ruleClass := range android.SortedStringKeys(metrics.ruleClassCount) {
65 count := metrics.ruleClassCount[ruleClass]
Jingwen Chen164e0862021-02-19 00:48:40 -050066 fmt.Printf("[bp2build] %s: %d targets\n", ruleClass, count)
67 generatedTargetCount += count
68 }
69 fmt.Printf(
Liz Kammerdaa09ef2021-12-15 15:35:38 -050070 `[bp2build] Converted %d Android.bp modules to %d total generated BUILD targets. Included %d handcrafted BUILD targets. There are %d total Android.bp modules.
71%d converted modules have unconverted deps:
72 %s
73%d converted modules have missing deps:
74 %s
75`,
Chris Parsons91b81f02021-12-08 11:19:06 -050076 metrics.generatedModuleCount,
Jingwen Chen164e0862021-02-19 00:48:40 -050077 generatedTargetCount,
Jingwen Chen310bc8f2021-09-20 10:54:27 +000078 metrics.handCraftedModuleCount,
79 metrics.TotalModuleCount(),
Liz Kammer6eff3232021-08-26 08:37:59 -040080 len(metrics.moduleWithUnconvertedDepsMsgs),
Liz Kammerdaa09ef2021-12-15 15:35:38 -050081 strings.Join(metrics.moduleWithUnconvertedDepsMsgs, "\n\t"),
82 len(metrics.moduleWithMissingDepsMsgs),
83 strings.Join(metrics.moduleWithMissingDepsMsgs, "\n\t"),
84 )
Jingwen Chen164e0862021-02-19 00:48:40 -050085}
Jingwen Chen61174502021-09-17 08:40:45 +000086
Alex Márquez Pérez Muñíz Díaz Púras Thaureaux947fdbf2021-11-10 09:55:20 -050087const bp2buildMetricsFilename = "bp2build_metrics.pb"
88
89// fail prints $PWD to stderr, followed by the given printf string and args (vals),
90// then the given alert, and then exits with 1 for failure
91func fail(err error, alertFmt string, vals ...interface{}) {
92 cwd, wderr := os.Getwd()
93 if wderr != nil {
94 cwd = "FAILED TO GET $PWD: " + wderr.Error()
95 }
96 fmt.Fprintf(os.Stderr, "\nIn "+cwd+":\n"+alertFmt+"\n"+err.Error()+"\n", vals...)
97 os.Exit(1)
98}
99
100// Write the bp2build-protoized codegen metrics into the given directory
101func (metrics *CodegenMetrics) Write(dir string) {
102 if _, err := os.Stat(dir); os.IsNotExist(err) {
103 // The metrics dir doesn't already exist, so create it (and parents)
104 if err := os.MkdirAll(dir, 0755); err != nil { // rx for all; w for user
105 fail(err, "Failed to `mkdir -p` %s", dir)
106 }
107 } else if err != nil {
108 fail(err, "Failed to `stat` %s", dir)
109 }
110 metricsFile := filepath.Join(dir, bp2buildMetricsFilename)
111 if err := metrics.dump(metricsFile); err != nil {
112 fail(err, "Error outputting %s", metricsFile)
113 }
114 if _, err := os.Stat(metricsFile); err != nil {
115 fail(err, "MISSING BP2BUILD METRICS OUTPUT: Failed to `stat` %s", metricsFile)
116 } else {
117 fmt.Printf("\nWrote bp2build metrics to: %s\n", metricsFile)
118 }
119}
120
Jingwen Chen310bc8f2021-09-20 10:54:27 +0000121func (metrics *CodegenMetrics) IncrementRuleClassCount(ruleClass string) {
122 metrics.ruleClassCount[ruleClass] += 1
123}
124
Chris Parsons492bd912022-01-20 12:55:05 -0500125func (metrics *CodegenMetrics) AddUnconvertedModule(moduleType string) {
Jingwen Chen310bc8f2021-09-20 10:54:27 +0000126 metrics.unconvertedModuleCount += 1
Chris Parsons492bd912022-01-20 12:55:05 -0500127 metrics.totalModuleTypeCount[moduleType] += 1
Jingwen Chen310bc8f2021-09-20 10:54:27 +0000128}
129
Alex Márquez Pérez Muñíz Díaz Púras Thaureaux947fdbf2021-11-10 09:55:20 -0500130func (metrics *CodegenMetrics) TotalModuleCount() uint64 {
Jingwen Chen310bc8f2021-09-20 10:54:27 +0000131 return metrics.handCraftedModuleCount +
132 metrics.generatedModuleCount +
133 metrics.unconvertedModuleCount
134}
135
Alex Márquez Pérez Muñíz Díaz Púras Thaureaux947fdbf2021-11-10 09:55:20 -0500136// Dump serializes the metrics to the given filename
137func (metrics *CodegenMetrics) dump(filename string) (err error) {
138 ser := metrics.Serialize()
139 return shared.Save(&ser, filename)
140}
141
Jingwen Chen310bc8f2021-09-20 10:54:27 +0000142type ConversionType int
143
144const (
145 Generated ConversionType = iota
146 Handcrafted
147)
148
Chris Parsons492bd912022-01-20 12:55:05 -0500149func (metrics *CodegenMetrics) AddConvertedModule(m blueprint.Module, moduleType string, conversionType ConversionType) {
Jingwen Chen61174502021-09-17 08:40:45 +0000150 // Undo prebuilt_ module name prefix modifications
Chris Parsons492bd912022-01-20 12:55:05 -0500151 moduleName := android.RemoveOptionalPrebuiltPrefix(m.Name())
Jingwen Chen61174502021-09-17 08:40:45 +0000152 metrics.convertedModules = append(metrics.convertedModules, moduleName)
Chris Parsons492bd912022-01-20 12:55:05 -0500153 metrics.convertedModuleTypeCount[moduleType] += 1
154 metrics.totalModuleTypeCount[moduleType] += 1
Jingwen Chen310bc8f2021-09-20 10:54:27 +0000155
156 if conversionType == Handcrafted {
157 metrics.handCraftedModuleCount += 1
158 } else if conversionType == Generated {
159 metrics.generatedModuleCount += 1
160 }
Jingwen Chen61174502021-09-17 08:40:45 +0000161}