blob: 68ac54435a523371167c40230a86caed1dbfe789 [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"
Jingwen Chen164e0862021-02-19 00:48:40 -050012)
13
14// Simple metrics struct to collect information about a Blueprint to BUILD
15// conversion process.
16type CodegenMetrics struct {
Jingwen Chen310bc8f2021-09-20 10:54:27 +000017 // 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 -050018 generatedModuleCount uint64
Jingwen Chen310bc8f2021-09-20 10:54:27 +000019
20 // 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 -050021 handCraftedModuleCount uint64
Jingwen Chen310bc8f2021-09-20 10:54:27 +000022
23 // Total number of unconverted Soong modules
Alex Márquez Pérez Muñíz Díaz Púras Thaureaux947fdbf2021-11-10 09:55:20 -050024 unconvertedModuleCount uint64
Jingwen Chen164e0862021-02-19 00:48:40 -050025
26 // 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 -050027 ruleClassCount map[string]uint64
Liz Kammer6eff3232021-08-26 08:37:59 -040028
Alex Márquez Pérez Muñíz Díaz Púras Thaureaux947fdbf2021-11-10 09:55:20 -050029 // List of modules with unconverted deps
30 // NOTE: NOT in the .proto
Liz Kammer6eff3232021-08-26 08:37:59 -040031 moduleWithUnconvertedDepsMsgs []string
Jingwen Chen61174502021-09-17 08:40:45 +000032
Alex Márquez Pérez Muñíz Díaz Púras Thaureaux947fdbf2021-11-10 09:55:20 -050033 // List of converted modules
Jingwen Chen61174502021-09-17 08:40:45 +000034 convertedModules []string
Jingwen Chen164e0862021-02-19 00:48:40 -050035}
36
Alex Márquez Pérez Muñíz Díaz Púras Thaureaux947fdbf2021-11-10 09:55:20 -050037// Serialize returns the protoized version of CodegenMetrics: bp2build_metrics_proto.Bp2BuildMetrics
38func (metrics *CodegenMetrics) Serialize() bp2build_metrics_proto.Bp2BuildMetrics {
39 return bp2build_metrics_proto.Bp2BuildMetrics{
40 GeneratedModuleCount: metrics.generatedModuleCount,
41 HandCraftedModuleCount: metrics.handCraftedModuleCount,
42 UnconvertedModuleCount: metrics.unconvertedModuleCount,
43 RuleClassCount: metrics.ruleClassCount,
44 ConvertedModules: metrics.convertedModules,
45 }
46}
47
Jingwen Chen164e0862021-02-19 00:48:40 -050048// Print the codegen metrics to stdout.
Jingwen Chenafb84bd2021-09-20 10:31:46 +000049func (metrics *CodegenMetrics) Print() {
Alex Márquez Pérez Muñíz Díaz Púras Thaureaux947fdbf2021-11-10 09:55:20 -050050 generatedTargetCount := uint64(0)
Jingwen Chen310bc8f2021-09-20 10:54:27 +000051 for _, ruleClass := range android.SortedStringKeys(metrics.ruleClassCount) {
52 count := metrics.ruleClassCount[ruleClass]
Jingwen Chen164e0862021-02-19 00:48:40 -050053 fmt.Printf("[bp2build] %s: %d targets\n", ruleClass, count)
54 generatedTargetCount += count
55 }
56 fmt.Printf(
Chris Parsons91b81f02021-12-08 11:19:06 -050057 "[bp2build] Converted %d Android.bp modules to %d total generated BUILD targets. Included %d handcrafted BUILD targets. There are %d total Android.bp modules.\n%d converted modules have unconverted deps: \n\t%s",
58 metrics.generatedModuleCount,
Jingwen Chen164e0862021-02-19 00:48:40 -050059 generatedTargetCount,
Jingwen Chen310bc8f2021-09-20 10:54:27 +000060 metrics.handCraftedModuleCount,
61 metrics.TotalModuleCount(),
Liz Kammer6eff3232021-08-26 08:37:59 -040062 len(metrics.moduleWithUnconvertedDepsMsgs),
63 strings.Join(metrics.moduleWithUnconvertedDepsMsgs, "\n\t"))
Jingwen Chen164e0862021-02-19 00:48:40 -050064}
Jingwen Chen61174502021-09-17 08:40:45 +000065
Alex Márquez Pérez Muñíz Díaz Púras Thaureaux947fdbf2021-11-10 09:55:20 -050066const bp2buildMetricsFilename = "bp2build_metrics.pb"
67
68// fail prints $PWD to stderr, followed by the given printf string and args (vals),
69// then the given alert, and then exits with 1 for failure
70func fail(err error, alertFmt string, vals ...interface{}) {
71 cwd, wderr := os.Getwd()
72 if wderr != nil {
73 cwd = "FAILED TO GET $PWD: " + wderr.Error()
74 }
75 fmt.Fprintf(os.Stderr, "\nIn "+cwd+":\n"+alertFmt+"\n"+err.Error()+"\n", vals...)
76 os.Exit(1)
77}
78
79// Write the bp2build-protoized codegen metrics into the given directory
80func (metrics *CodegenMetrics) Write(dir string) {
81 if _, err := os.Stat(dir); os.IsNotExist(err) {
82 // The metrics dir doesn't already exist, so create it (and parents)
83 if err := os.MkdirAll(dir, 0755); err != nil { // rx for all; w for user
84 fail(err, "Failed to `mkdir -p` %s", dir)
85 }
86 } else if err != nil {
87 fail(err, "Failed to `stat` %s", dir)
88 }
89 metricsFile := filepath.Join(dir, bp2buildMetricsFilename)
90 if err := metrics.dump(metricsFile); err != nil {
91 fail(err, "Error outputting %s", metricsFile)
92 }
93 if _, err := os.Stat(metricsFile); err != nil {
94 fail(err, "MISSING BP2BUILD METRICS OUTPUT: Failed to `stat` %s", metricsFile)
95 } else {
96 fmt.Printf("\nWrote bp2build metrics to: %s\n", metricsFile)
97 }
98}
99
Jingwen Chen310bc8f2021-09-20 10:54:27 +0000100func (metrics *CodegenMetrics) IncrementRuleClassCount(ruleClass string) {
101 metrics.ruleClassCount[ruleClass] += 1
102}
103
104func (metrics *CodegenMetrics) IncrementUnconvertedCount() {
105 metrics.unconvertedModuleCount += 1
106}
107
Alex Márquez Pérez Muñíz Díaz Púras Thaureaux947fdbf2021-11-10 09:55:20 -0500108func (metrics *CodegenMetrics) TotalModuleCount() uint64 {
Jingwen Chen310bc8f2021-09-20 10:54:27 +0000109 return metrics.handCraftedModuleCount +
110 metrics.generatedModuleCount +
111 metrics.unconvertedModuleCount
112}
113
Alex Márquez Pérez Muñíz Díaz Púras Thaureaux947fdbf2021-11-10 09:55:20 -0500114// Dump serializes the metrics to the given filename
115func (metrics *CodegenMetrics) dump(filename string) (err error) {
116 ser := metrics.Serialize()
117 return shared.Save(&ser, filename)
118}
119
Jingwen Chen310bc8f2021-09-20 10:54:27 +0000120type ConversionType int
121
122const (
123 Generated ConversionType = iota
124 Handcrafted
125)
126
127func (metrics *CodegenMetrics) AddConvertedModule(moduleName string, conversionType ConversionType) {
Jingwen Chen61174502021-09-17 08:40:45 +0000128 // Undo prebuilt_ module name prefix modifications
129 moduleName = android.RemoveOptionalPrebuiltPrefix(moduleName)
130 metrics.convertedModules = append(metrics.convertedModules, moduleName)
Jingwen Chen310bc8f2021-09-20 10:54:27 +0000131
132 if conversionType == Handcrafted {
133 metrics.handCraftedModuleCount += 1
134 } else if conversionType == Generated {
135 metrics.generatedModuleCount += 1
136 }
Jingwen Chen61174502021-09-17 08:40:45 +0000137}