1*333d2b36SAndroid Build Coastguard Worker// Copyright 2017 Google Inc. All rights reserved. 2*333d2b36SAndroid Build Coastguard Worker// 3*333d2b36SAndroid Build Coastguard Worker// Licensed under the Apache License, Version 2.0 (the "License"); 4*333d2b36SAndroid Build Coastguard Worker// you may not use this file except in compliance with the License. 5*333d2b36SAndroid Build Coastguard Worker// You may obtain a copy of the License at 6*333d2b36SAndroid Build Coastguard Worker// 7*333d2b36SAndroid Build Coastguard Worker// http://www.apache.org/licenses/LICENSE-2.0 8*333d2b36SAndroid Build Coastguard Worker// 9*333d2b36SAndroid Build Coastguard Worker// Unless required by applicable law or agreed to in writing, software 10*333d2b36SAndroid Build Coastguard Worker// distributed under the License is distributed on an "AS IS" BASIS, 11*333d2b36SAndroid Build Coastguard Worker// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 12*333d2b36SAndroid Build Coastguard Worker// See the License for the specific language governing permissions and 13*333d2b36SAndroid Build Coastguard Worker// limitations under the License. 14*333d2b36SAndroid Build Coastguard Worker 15*333d2b36SAndroid Build Coastguard Workerpackage jar 16*333d2b36SAndroid Build Coastguard Worker 17*333d2b36SAndroid Build Coastguard Workerimport ( 18*333d2b36SAndroid Build Coastguard Worker "bytes" 19*333d2b36SAndroid Build Coastguard Worker "fmt" 20*333d2b36SAndroid Build Coastguard Worker "io" 21*333d2b36SAndroid Build Coastguard Worker "os" 22*333d2b36SAndroid Build Coastguard Worker "strings" 23*333d2b36SAndroid Build Coastguard Worker "text/scanner" 24*333d2b36SAndroid Build Coastguard Worker "time" 25*333d2b36SAndroid Build Coastguard Worker "unicode" 26*333d2b36SAndroid Build Coastguard Worker 27*333d2b36SAndroid Build Coastguard Worker "android/soong/third_party/zip" 28*333d2b36SAndroid Build Coastguard Worker) 29*333d2b36SAndroid Build Coastguard Worker 30*333d2b36SAndroid Build Coastguard Workerconst ( 31*333d2b36SAndroid Build Coastguard Worker MetaDir = "META-INF/" 32*333d2b36SAndroid Build Coastguard Worker ManifestFile = MetaDir + "MANIFEST.MF" 33*333d2b36SAndroid Build Coastguard Worker ModuleInfoClass = "module-info.class" 34*333d2b36SAndroid Build Coastguard Worker) 35*333d2b36SAndroid Build Coastguard Worker 36*333d2b36SAndroid Build Coastguard Workervar DefaultTime = time.Date(2008, 1, 1, 0, 0, 0, 0, time.UTC) 37*333d2b36SAndroid Build Coastguard Worker 38*333d2b36SAndroid Build Coastguard Workervar MetaDirExtra = [2]byte{0xca, 0xfe} 39*333d2b36SAndroid Build Coastguard Worker 40*333d2b36SAndroid Build Coastguard Worker// EntryNamesLess tells whether <filepathA> should precede <filepathB> in 41*333d2b36SAndroid Build Coastguard Worker// the order of files with a .jar 42*333d2b36SAndroid Build Coastguard Workerfunc EntryNamesLess(filepathA string, filepathB string) (less bool) { 43*333d2b36SAndroid Build Coastguard Worker diff := index(filepathA) - index(filepathB) 44*333d2b36SAndroid Build Coastguard Worker if diff == 0 { 45*333d2b36SAndroid Build Coastguard Worker return filepathA < filepathB 46*333d2b36SAndroid Build Coastguard Worker } 47*333d2b36SAndroid Build Coastguard Worker return diff < 0 48*333d2b36SAndroid Build Coastguard Worker} 49*333d2b36SAndroid Build Coastguard Worker 50*333d2b36SAndroid Build Coastguard Worker// Treats trailing * as a prefix match 51*333d2b36SAndroid Build Coastguard Workerfunc patternMatch(pattern, name string) bool { 52*333d2b36SAndroid Build Coastguard Worker if strings.HasSuffix(pattern, "*") { 53*333d2b36SAndroid Build Coastguard Worker return strings.HasPrefix(name, strings.TrimSuffix(pattern, "*")) 54*333d2b36SAndroid Build Coastguard Worker } else { 55*333d2b36SAndroid Build Coastguard Worker return name == pattern 56*333d2b36SAndroid Build Coastguard Worker } 57*333d2b36SAndroid Build Coastguard Worker} 58*333d2b36SAndroid Build Coastguard Worker 59*333d2b36SAndroid Build Coastguard Workervar jarOrder = []string{ 60*333d2b36SAndroid Build Coastguard Worker MetaDir, 61*333d2b36SAndroid Build Coastguard Worker ManifestFile, 62*333d2b36SAndroid Build Coastguard Worker MetaDir + "*", 63*333d2b36SAndroid Build Coastguard Worker "*", 64*333d2b36SAndroid Build Coastguard Worker} 65*333d2b36SAndroid Build Coastguard Worker 66*333d2b36SAndroid Build Coastguard Workerfunc index(name string) int { 67*333d2b36SAndroid Build Coastguard Worker for i, pattern := range jarOrder { 68*333d2b36SAndroid Build Coastguard Worker if patternMatch(pattern, name) { 69*333d2b36SAndroid Build Coastguard Worker return i 70*333d2b36SAndroid Build Coastguard Worker } 71*333d2b36SAndroid Build Coastguard Worker } 72*333d2b36SAndroid Build Coastguard Worker panic(fmt.Errorf("file %q did not match any pattern", name)) 73*333d2b36SAndroid Build Coastguard Worker} 74*333d2b36SAndroid Build Coastguard Worker 75*333d2b36SAndroid Build Coastguard Workerfunc MetaDirFileHeader() *zip.FileHeader { 76*333d2b36SAndroid Build Coastguard Worker dirHeader := &zip.FileHeader{ 77*333d2b36SAndroid Build Coastguard Worker Name: MetaDir, 78*333d2b36SAndroid Build Coastguard Worker Extra: []byte{MetaDirExtra[1], MetaDirExtra[0], 0, 0}, 79*333d2b36SAndroid Build Coastguard Worker } 80*333d2b36SAndroid Build Coastguard Worker dirHeader.SetMode(0755 | os.ModeDir) 81*333d2b36SAndroid Build Coastguard Worker dirHeader.SetModTime(DefaultTime) 82*333d2b36SAndroid Build Coastguard Worker 83*333d2b36SAndroid Build Coastguard Worker return dirHeader 84*333d2b36SAndroid Build Coastguard Worker} 85*333d2b36SAndroid Build Coastguard Worker 86*333d2b36SAndroid Build Coastguard Worker// Create a manifest zip header and contents using the provided contents if any. 87*333d2b36SAndroid Build Coastguard Workerfunc ManifestFileContents(contents []byte) (*zip.FileHeader, []byte, error) { 88*333d2b36SAndroid Build Coastguard Worker b, err := manifestContents(contents) 89*333d2b36SAndroid Build Coastguard Worker if err != nil { 90*333d2b36SAndroid Build Coastguard Worker return nil, nil, err 91*333d2b36SAndroid Build Coastguard Worker } 92*333d2b36SAndroid Build Coastguard Worker 93*333d2b36SAndroid Build Coastguard Worker fh := &zip.FileHeader{ 94*333d2b36SAndroid Build Coastguard Worker Name: ManifestFile, 95*333d2b36SAndroid Build Coastguard Worker Method: zip.Store, 96*333d2b36SAndroid Build Coastguard Worker UncompressedSize64: uint64(len(b)), 97*333d2b36SAndroid Build Coastguard Worker } 98*333d2b36SAndroid Build Coastguard Worker fh.SetMode(0644) 99*333d2b36SAndroid Build Coastguard Worker fh.SetModTime(DefaultTime) 100*333d2b36SAndroid Build Coastguard Worker 101*333d2b36SAndroid Build Coastguard Worker return fh, b, nil 102*333d2b36SAndroid Build Coastguard Worker} 103*333d2b36SAndroid Build Coastguard Worker 104*333d2b36SAndroid Build Coastguard Worker// Create manifest contents, using the provided contents if any. 105*333d2b36SAndroid Build Coastguard Workerfunc manifestContents(contents []byte) ([]byte, error) { 106*333d2b36SAndroid Build Coastguard Worker manifestMarker := []byte("Manifest-Version:") 107*333d2b36SAndroid Build Coastguard Worker header := append(manifestMarker, []byte(" 1.0\nCreated-By: soong_zip\n")...) 108*333d2b36SAndroid Build Coastguard Worker 109*333d2b36SAndroid Build Coastguard Worker var finalBytes []byte 110*333d2b36SAndroid Build Coastguard Worker if !bytes.Contains(contents, manifestMarker) { 111*333d2b36SAndroid Build Coastguard Worker finalBytes = append(append(header, contents...), byte('\n')) 112*333d2b36SAndroid Build Coastguard Worker } else { 113*333d2b36SAndroid Build Coastguard Worker finalBytes = contents 114*333d2b36SAndroid Build Coastguard Worker } 115*333d2b36SAndroid Build Coastguard Worker 116*333d2b36SAndroid Build Coastguard Worker return finalBytes, nil 117*333d2b36SAndroid Build Coastguard Worker} 118*333d2b36SAndroid Build Coastguard Worker 119*333d2b36SAndroid Build Coastguard Workervar javaIgnorableIdentifier = &unicode.RangeTable{ 120*333d2b36SAndroid Build Coastguard Worker R16: []unicode.Range16{ 121*333d2b36SAndroid Build Coastguard Worker {0x00, 0x08, 1}, 122*333d2b36SAndroid Build Coastguard Worker {0x0e, 0x1b, 1}, 123*333d2b36SAndroid Build Coastguard Worker {0x7f, 0x9f, 1}, 124*333d2b36SAndroid Build Coastguard Worker }, 125*333d2b36SAndroid Build Coastguard Worker LatinOffset: 3, 126*333d2b36SAndroid Build Coastguard Worker} 127*333d2b36SAndroid Build Coastguard Worker 128*333d2b36SAndroid Build Coastguard Workerfunc javaIdentRune(ch rune, i int) bool { 129*333d2b36SAndroid Build Coastguard Worker if unicode.IsLetter(ch) { 130*333d2b36SAndroid Build Coastguard Worker return true 131*333d2b36SAndroid Build Coastguard Worker } 132*333d2b36SAndroid Build Coastguard Worker if unicode.IsDigit(ch) && i > 0 { 133*333d2b36SAndroid Build Coastguard Worker return true 134*333d2b36SAndroid Build Coastguard Worker } 135*333d2b36SAndroid Build Coastguard Worker 136*333d2b36SAndroid Build Coastguard Worker if unicode.In(ch, 137*333d2b36SAndroid Build Coastguard Worker unicode.Nl, // letter number 138*333d2b36SAndroid Build Coastguard Worker unicode.Sc, // currency symbol 139*333d2b36SAndroid Build Coastguard Worker unicode.Pc, // connecting punctuation 140*333d2b36SAndroid Build Coastguard Worker ) { 141*333d2b36SAndroid Build Coastguard Worker return true 142*333d2b36SAndroid Build Coastguard Worker } 143*333d2b36SAndroid Build Coastguard Worker 144*333d2b36SAndroid Build Coastguard Worker if unicode.In(ch, 145*333d2b36SAndroid Build Coastguard Worker unicode.Cf, // format 146*333d2b36SAndroid Build Coastguard Worker unicode.Mc, // combining mark 147*333d2b36SAndroid Build Coastguard Worker unicode.Mn, // non-spacing mark 148*333d2b36SAndroid Build Coastguard Worker javaIgnorableIdentifier, 149*333d2b36SAndroid Build Coastguard Worker ) && i > 0 { 150*333d2b36SAndroid Build Coastguard Worker return true 151*333d2b36SAndroid Build Coastguard Worker } 152*333d2b36SAndroid Build Coastguard Worker 153*333d2b36SAndroid Build Coastguard Worker return false 154*333d2b36SAndroid Build Coastguard Worker} 155*333d2b36SAndroid Build Coastguard Worker 156*333d2b36SAndroid Build Coastguard Worker// JavaPackage parses the package out of a java source file by looking for the package statement, or the first valid 157*333d2b36SAndroid Build Coastguard Worker// non-package statement, in which case it returns an empty string for the package. 158*333d2b36SAndroid Build Coastguard Workerfunc JavaPackage(r io.Reader, src string) (string, error) { 159*333d2b36SAndroid Build Coastguard Worker var s scanner.Scanner 160*333d2b36SAndroid Build Coastguard Worker var sErr error 161*333d2b36SAndroid Build Coastguard Worker 162*333d2b36SAndroid Build Coastguard Worker s.Init(r) 163*333d2b36SAndroid Build Coastguard Worker s.Filename = src 164*333d2b36SAndroid Build Coastguard Worker s.Error = func(s *scanner.Scanner, msg string) { 165*333d2b36SAndroid Build Coastguard Worker sErr = fmt.Errorf("error parsing %q: %s", src, msg) 166*333d2b36SAndroid Build Coastguard Worker } 167*333d2b36SAndroid Build Coastguard Worker s.IsIdentRune = javaIdentRune 168*333d2b36SAndroid Build Coastguard Worker 169*333d2b36SAndroid Build Coastguard Worker var tok rune 170*333d2b36SAndroid Build Coastguard Worker for { 171*333d2b36SAndroid Build Coastguard Worker tok = s.Scan() 172*333d2b36SAndroid Build Coastguard Worker if sErr != nil { 173*333d2b36SAndroid Build Coastguard Worker return "", sErr 174*333d2b36SAndroid Build Coastguard Worker } 175*333d2b36SAndroid Build Coastguard Worker // If the first token is an annotation, it could be annotating a package declaration, so consume them. 176*333d2b36SAndroid Build Coastguard Worker // Note that this does not support "complex" annotations with attributes, e.g. @Foo(x=y). 177*333d2b36SAndroid Build Coastguard Worker if tok != '@' { 178*333d2b36SAndroid Build Coastguard Worker break 179*333d2b36SAndroid Build Coastguard Worker } 180*333d2b36SAndroid Build Coastguard Worker tok = s.Scan() 181*333d2b36SAndroid Build Coastguard Worker if tok != scanner.Ident || sErr != nil { 182*333d2b36SAndroid Build Coastguard Worker return "", fmt.Errorf("expected annotation identifier, got @%v", tok) 183*333d2b36SAndroid Build Coastguard Worker } 184*333d2b36SAndroid Build Coastguard Worker } 185*333d2b36SAndroid Build Coastguard Worker 186*333d2b36SAndroid Build Coastguard Worker if tok == scanner.Ident { 187*333d2b36SAndroid Build Coastguard Worker switch s.TokenText() { 188*333d2b36SAndroid Build Coastguard Worker case "package": 189*333d2b36SAndroid Build Coastguard Worker // Nothing 190*333d2b36SAndroid Build Coastguard Worker case "import": 191*333d2b36SAndroid Build Coastguard Worker // File has no package statement, first keyword is an import 192*333d2b36SAndroid Build Coastguard Worker return "", nil 193*333d2b36SAndroid Build Coastguard Worker case "class", "enum", "interface": 194*333d2b36SAndroid Build Coastguard Worker // File has no package statement, first keyword is a type declaration 195*333d2b36SAndroid Build Coastguard Worker return "", nil 196*333d2b36SAndroid Build Coastguard Worker case "public", "protected", "private", "abstract", "static", "final", "strictfp": 197*333d2b36SAndroid Build Coastguard Worker // File has no package statement, first keyword is a modifier 198*333d2b36SAndroid Build Coastguard Worker return "", nil 199*333d2b36SAndroid Build Coastguard Worker case "module", "open": 200*333d2b36SAndroid Build Coastguard Worker // File has no package statement, first keyword is a module declaration 201*333d2b36SAndroid Build Coastguard Worker return "", nil 202*333d2b36SAndroid Build Coastguard Worker default: 203*333d2b36SAndroid Build Coastguard Worker return "", fmt.Errorf(`expected first token of java file to be "package", got %q`, s.TokenText()) 204*333d2b36SAndroid Build Coastguard Worker } 205*333d2b36SAndroid Build Coastguard Worker } else if tok == scanner.EOF { 206*333d2b36SAndroid Build Coastguard Worker // File no package statement, it has no non-whitespace non-comment tokens 207*333d2b36SAndroid Build Coastguard Worker return "", nil 208*333d2b36SAndroid Build Coastguard Worker } else { 209*333d2b36SAndroid Build Coastguard Worker return "", fmt.Errorf(`expected first token of java file to be "package", got %q`, s.TokenText()) 210*333d2b36SAndroid Build Coastguard Worker } 211*333d2b36SAndroid Build Coastguard Worker 212*333d2b36SAndroid Build Coastguard Worker var pkg string 213*333d2b36SAndroid Build Coastguard Worker for { 214*333d2b36SAndroid Build Coastguard Worker tok = s.Scan() 215*333d2b36SAndroid Build Coastguard Worker if sErr != nil { 216*333d2b36SAndroid Build Coastguard Worker return "", sErr 217*333d2b36SAndroid Build Coastguard Worker } 218*333d2b36SAndroid Build Coastguard Worker if tok != scanner.Ident { 219*333d2b36SAndroid Build Coastguard Worker return "", fmt.Errorf(`expected "package <package>;", got "package %s%s"`, pkg, s.TokenText()) 220*333d2b36SAndroid Build Coastguard Worker } 221*333d2b36SAndroid Build Coastguard Worker pkg += s.TokenText() 222*333d2b36SAndroid Build Coastguard Worker 223*333d2b36SAndroid Build Coastguard Worker tok = s.Scan() 224*333d2b36SAndroid Build Coastguard Worker if sErr != nil { 225*333d2b36SAndroid Build Coastguard Worker return "", sErr 226*333d2b36SAndroid Build Coastguard Worker } 227*333d2b36SAndroid Build Coastguard Worker if tok == ';' { 228*333d2b36SAndroid Build Coastguard Worker return pkg, nil 229*333d2b36SAndroid Build Coastguard Worker } else if tok == '.' { 230*333d2b36SAndroid Build Coastguard Worker pkg += "." 231*333d2b36SAndroid Build Coastguard Worker } else { 232*333d2b36SAndroid Build Coastguard Worker return "", fmt.Errorf(`expected "package <package>;", got "package %s%s"`, pkg, s.TokenText()) 233*333d2b36SAndroid Build Coastguard Worker } 234*333d2b36SAndroid Build Coastguard Worker } 235*333d2b36SAndroid Build Coastguard Worker} 236