1
2
3
4
5
6 package cache
7
8 import (
9 "bytes"
10 "crypto/sha256"
11 "encoding/hex"
12 "errors"
13 "fmt"
14 "internal/godebug"
15 "io"
16 "io/fs"
17 "os"
18 "path/filepath"
19 "strconv"
20 "strings"
21 "time"
22
23 "cmd/go/internal/base"
24 "cmd/go/internal/lockedfile"
25 "cmd/go/internal/mmap"
26 )
27
28
29
30
31 type ActionID [HashSize]byte
32
33
34 type OutputID [HashSize]byte
35
36
37 type Cache interface {
38
39
40
41
42
43 Get(ActionID) (Entry, error)
44
45
46
47
48
49
50
51
52
53
54
55 Put(ActionID, io.ReadSeeker) (_ OutputID, size int64, _ error)
56
57
58
59
60
61
62
63
64
65 Close() error
66
67
68
69
70
71
72 OutputFile(OutputID) string
73
74
75 FuzzDir() string
76 }
77
78
79 type DiskCache struct {
80 dir string
81 now func() time.Time
82 }
83
84
85
86
87
88
89
90
91
92
93
94
95 func Open(dir string) (*DiskCache, error) {
96 info, err := os.Stat(dir)
97 if err != nil {
98 return nil, err
99 }
100 if !info.IsDir() {
101 return nil, &fs.PathError{Op: "open", Path: dir, Err: fmt.Errorf("not a directory")}
102 }
103 for i := 0; i < 256; i++ {
104 name := filepath.Join(dir, fmt.Sprintf("%02x", i))
105 if err := os.MkdirAll(name, 0o777); err != nil {
106 return nil, err
107 }
108 }
109 c := &DiskCache{
110 dir: dir,
111 now: time.Now,
112 }
113 return c, nil
114 }
115
116
117 func (c *DiskCache) fileName(id [HashSize]byte, key string) string {
118 return filepath.Join(c.dir, fmt.Sprintf("%02x", id[0]), fmt.Sprintf("%x", id)+"-"+key)
119 }
120
121
122
123 type entryNotFoundError struct {
124 Err error
125 }
126
127 func (e *entryNotFoundError) Error() string {
128 if e.Err == nil {
129 return "cache entry not found"
130 }
131 return fmt.Sprintf("cache entry not found: %v", e.Err)
132 }
133
134 func (e *entryNotFoundError) Unwrap() error {
135 return e.Err
136 }
137
138 const (
139
140 hexSize = HashSize * 2
141 entrySize = 2 + 1 + hexSize + 1 + hexSize + 1 + 20 + 1 + 20 + 1
142 )
143
144
145
146
147
148
149
150
151
152
153 var verify = false
154
155 var errVerifyMode = errors.New("gocacheverify=1")
156
157
158 var DebugTest = false
159
160 func init() { initEnv() }
161
162 var (
163 gocacheverify = godebug.New("gocacheverify")
164 gocachehash = godebug.New("gocachehash")
165 gocachetest = godebug.New("gocachetest")
166 )
167
168 func initEnv() {
169 if gocacheverify.Value() == "1" {
170 gocacheverify.IncNonDefault()
171 verify = true
172 }
173 if gocachehash.Value() == "1" {
174 gocachehash.IncNonDefault()
175 debugHash = true
176 }
177 if gocachetest.Value() == "1" {
178 gocachetest.IncNonDefault()
179 DebugTest = true
180 }
181 }
182
183
184
185
186
187 func (c *DiskCache) Get(id ActionID) (Entry, error) {
188 if verify {
189 return Entry{}, &entryNotFoundError{Err: errVerifyMode}
190 }
191 return c.get(id)
192 }
193
194 type Entry struct {
195 OutputID OutputID
196 Size int64
197 Time time.Time
198 }
199
200
201 func (c *DiskCache) get(id ActionID) (Entry, error) {
202 missing := func(reason error) (Entry, error) {
203 return Entry{}, &entryNotFoundError{Err: reason}
204 }
205 f, err := os.Open(c.fileName(id, "a"))
206 if err != nil {
207 return missing(err)
208 }
209 defer f.Close()
210 entry := make([]byte, entrySize+1)
211 if n, err := io.ReadFull(f, entry); n > entrySize {
212 return missing(errors.New("too long"))
213 } else if err != io.ErrUnexpectedEOF {
214 if err == io.EOF {
215 return missing(errors.New("file is empty"))
216 }
217 return missing(err)
218 } else if n < entrySize {
219 return missing(errors.New("entry file incomplete"))
220 }
221 if entry[0] != 'v' || entry[1] != '1' || entry[2] != ' ' || entry[3+hexSize] != ' ' || entry[3+hexSize+1+hexSize] != ' ' || entry[3+hexSize+1+hexSize+1+20] != ' ' || entry[entrySize-1] != '\n' {
222 return missing(errors.New("invalid header"))
223 }
224 eid, entry := entry[3:3+hexSize], entry[3+hexSize:]
225 eout, entry := entry[1:1+hexSize], entry[1+hexSize:]
226 esize, entry := entry[1:1+20], entry[1+20:]
227 etime, entry := entry[1:1+20], entry[1+20:]
228 var buf [HashSize]byte
229 if _, err := hex.Decode(buf[:], eid); err != nil {
230 return missing(fmt.Errorf("decoding ID: %v", err))
231 } else if buf != id {
232 return missing(errors.New("mismatched ID"))
233 }
234 if _, err := hex.Decode(buf[:], eout); err != nil {
235 return missing(fmt.Errorf("decoding output ID: %v", err))
236 }
237 i := 0
238 for i < len(esize) && esize[i] == ' ' {
239 i++
240 }
241 size, err := strconv.ParseInt(string(esize[i:]), 10, 64)
242 if err != nil {
243 return missing(fmt.Errorf("parsing size: %v", err))
244 } else if size < 0 {
245 return missing(errors.New("negative size"))
246 }
247 i = 0
248 for i < len(etime) && etime[i] == ' ' {
249 i++
250 }
251 tm, err := strconv.ParseInt(string(etime[i:]), 10, 64)
252 if err != nil {
253 return missing(fmt.Errorf("parsing timestamp: %v", err))
254 } else if tm < 0 {
255 return missing(errors.New("negative timestamp"))
256 }
257
258 c.markUsed(c.fileName(id, "a"))
259
260 return Entry{buf, size, time.Unix(0, tm)}, nil
261 }
262
263
264
265 func GetFile(c Cache, id ActionID) (file string, entry Entry, err error) {
266 entry, err = c.Get(id)
267 if err != nil {
268 return "", Entry{}, err
269 }
270 file = c.OutputFile(entry.OutputID)
271 info, err := os.Stat(file)
272 if err != nil {
273 return "", Entry{}, &entryNotFoundError{Err: err}
274 }
275 if info.Size() != entry.Size {
276 return "", Entry{}, &entryNotFoundError{Err: errors.New("file incomplete")}
277 }
278 return file, entry, nil
279 }
280
281
282
283
284 func GetBytes(c Cache, id ActionID) ([]byte, Entry, error) {
285 entry, err := c.Get(id)
286 if err != nil {
287 return nil, entry, err
288 }
289 data, _ := os.ReadFile(c.OutputFile(entry.OutputID))
290 if sha256.Sum256(data) != entry.OutputID {
291 return nil, entry, &entryNotFoundError{Err: errors.New("bad checksum")}
292 }
293 return data, entry, nil
294 }
295
296
297
298
299 func GetMmap(c Cache, id ActionID) ([]byte, Entry, bool, error) {
300 entry, err := c.Get(id)
301 if err != nil {
302 return nil, entry, false, err
303 }
304 md, opened, err := mmap.Mmap(c.OutputFile(entry.OutputID))
305 if err != nil {
306 return nil, Entry{}, opened, err
307 }
308 if int64(len(md.Data)) != entry.Size {
309 return nil, Entry{}, true, &entryNotFoundError{Err: errors.New("file incomplete")}
310 }
311 return md.Data, entry, true, nil
312 }
313
314
315 func (c *DiskCache) OutputFile(out OutputID) string {
316 file := c.fileName(out, "d")
317 isDir := c.markUsed(file)
318 if isDir {
319 entries, err := os.ReadDir(file)
320 if err != nil {
321 return fmt.Sprintf("DO NOT USE - missing binary cache entry: %v", err)
322 }
323 if len(entries) != 1 {
324 return "DO NOT USE - invalid binary cache entry"
325 }
326 return filepath.Join(file, entries[0].Name())
327 }
328 return file
329 }
330
331
332
333
334
335
336
337
338
339
340
341
342
343 const (
344 mtimeInterval = 1 * time.Hour
345 trimInterval = 24 * time.Hour
346 trimLimit = 5 * 24 * time.Hour
347 )
348
349
350
351
352
353
354
355
356
357
358
359
360 func (c *DiskCache) markUsed(file string) (isDir bool) {
361 info, err := os.Stat(file)
362 if err != nil {
363 return false
364 }
365 if now := c.now(); now.Sub(info.ModTime()) >= mtimeInterval {
366 os.Chtimes(file, now, now)
367 }
368 return info.IsDir()
369 }
370
371 func (c *DiskCache) Close() error { return c.Trim() }
372
373
374 func (c *DiskCache) Trim() error {
375 now := c.now()
376
377
378
379
380
381
382
383
384 if data, err := lockedfile.Read(filepath.Join(c.dir, "trim.txt")); err == nil {
385 if t, err := strconv.ParseInt(strings.TrimSpace(string(data)), 10, 64); err == nil {
386 lastTrim := time.Unix(t, 0)
387 if d := now.Sub(lastTrim); d < trimInterval && d > -mtimeInterval {
388 return nil
389 }
390 }
391 }
392
393
394
395
396 cutoff := now.Add(-trimLimit - mtimeInterval)
397 for i := 0; i < 256; i++ {
398 subdir := filepath.Join(c.dir, fmt.Sprintf("%02x", i))
399 c.trimSubdir(subdir, cutoff)
400 }
401
402
403
404 var b bytes.Buffer
405 fmt.Fprintf(&b, "%d", now.Unix())
406 if err := lockedfile.Write(filepath.Join(c.dir, "trim.txt"), &b, 0o666); err != nil {
407 return err
408 }
409
410 return nil
411 }
412
413
414 func (c *DiskCache) trimSubdir(subdir string, cutoff time.Time) {
415
416
417
418
419
420 f, err := os.Open(subdir)
421 if err != nil {
422 return
423 }
424 names, _ := f.Readdirnames(-1)
425 f.Close()
426
427 for _, name := range names {
428
429 if !strings.HasSuffix(name, "-a") && !strings.HasSuffix(name, "-d") {
430 continue
431 }
432 entry := filepath.Join(subdir, name)
433 info, err := os.Stat(entry)
434 if err == nil && info.ModTime().Before(cutoff) {
435 if info.IsDir() {
436 os.RemoveAll(entry)
437 continue
438 }
439 os.Remove(entry)
440 }
441 }
442 }
443
444
445
446 func (c *DiskCache) putIndexEntry(id ActionID, out OutputID, size int64, allowVerify bool) error {
447
448
449
450
451
452
453
454
455
456
457
458 entry := fmt.Sprintf("v1 %x %x %20d %20d\n", id, out, size, time.Now().UnixNano())
459 if verify && allowVerify {
460 old, err := c.get(id)
461 if err == nil && (old.OutputID != out || old.Size != size) {
462
463 msg := fmt.Sprintf("go: internal cache error: cache verify failed: id=%x changed:<<<\n%s\n>>>\nold: %x %d\nnew: %x %d", id, reverseHash(id), out, size, old.OutputID, old.Size)
464 panic(msg)
465 }
466 }
467 file := c.fileName(id, "a")
468
469
470 mode := os.O_WRONLY | os.O_CREATE
471 f, err := os.OpenFile(file, mode, 0o666)
472 if err != nil {
473 return err
474 }
475 _, err = f.WriteString(entry)
476 if err == nil {
477
478
479
480
481
482
483
484 err = f.Truncate(int64(len(entry)))
485 }
486 if closeErr := f.Close(); err == nil {
487 err = closeErr
488 }
489 if err != nil {
490
491
492 os.Remove(file)
493 return err
494 }
495 os.Chtimes(file, c.now(), c.now())
496
497 return nil
498 }
499
500
501
502
503 type noVerifyReadSeeker struct {
504 io.ReadSeeker
505 }
506
507
508
509 func (c *DiskCache) Put(id ActionID, file io.ReadSeeker) (OutputID, int64, error) {
510 wrapper, isNoVerify := file.(noVerifyReadSeeker)
511 if isNoVerify {
512 file = wrapper.ReadSeeker
513 }
514 return c.put(id, "", file, !isNoVerify)
515 }
516
517
518
519
520 func (c *DiskCache) PutExecutable(id ActionID, name string, file io.ReadSeeker) (OutputID, int64, error) {
521 if name == "" {
522 panic("PutExecutable called without a name")
523 }
524 wrapper, isNoVerify := file.(noVerifyReadSeeker)
525 if isNoVerify {
526 file = wrapper.ReadSeeker
527 }
528 return c.put(id, name, file, !isNoVerify)
529 }
530
531
532
533
534
535 func PutNoVerify(c Cache, id ActionID, file io.ReadSeeker) (OutputID, int64, error) {
536 return c.Put(id, noVerifyReadSeeker{file})
537 }
538
539 func (c *DiskCache) put(id ActionID, executableName string, file io.ReadSeeker, allowVerify bool) (OutputID, int64, error) {
540
541 h := sha256.New()
542 if _, err := file.Seek(0, 0); err != nil {
543 return OutputID{}, 0, err
544 }
545 size, err := io.Copy(h, file)
546 if err != nil {
547 return OutputID{}, 0, err
548 }
549 var out OutputID
550 h.Sum(out[:0])
551
552
553 fileMode := fs.FileMode(0o666)
554 if executableName != "" {
555 fileMode = 0o777
556 }
557 if err := c.copyFile(file, executableName, out, size, fileMode); err != nil {
558 return out, size, err
559 }
560
561
562 return out, size, c.putIndexEntry(id, out, size, allowVerify)
563 }
564
565
566 func PutBytes(c Cache, id ActionID, data []byte) error {
567 _, _, err := c.Put(id, bytes.NewReader(data))
568 return err
569 }
570
571
572
573 func (c *DiskCache) copyFile(file io.ReadSeeker, executableName string, out OutputID, size int64, perm os.FileMode) error {
574 name := c.fileName(out, "d")
575 info, err := os.Stat(name)
576 if executableName != "" {
577
578
579
580
581 if err != nil {
582 if !os.IsNotExist(err) {
583 return err
584 }
585 if err := os.Mkdir(name, 0o777); err != nil {
586 return err
587 }
588 if info, err = os.Stat(name); err != nil {
589 return err
590 }
591 }
592 if !info.IsDir() {
593 return errors.New("internal error: invalid binary cache entry: not a directory")
594 }
595
596
597 name = filepath.Join(name, executableName)
598 info, err = os.Stat(name)
599 }
600 if err == nil && info.Size() == size {
601
602 if f, err := os.Open(name); err == nil {
603 h := sha256.New()
604 io.Copy(h, f)
605 f.Close()
606 var out2 OutputID
607 h.Sum(out2[:0])
608 if out == out2 {
609 return nil
610 }
611 }
612
613 }
614
615
616 mode := os.O_RDWR | os.O_CREATE
617 if err == nil && info.Size() > size {
618 mode |= os.O_TRUNC
619 }
620 f, err := os.OpenFile(name, mode, perm)
621 if err != nil {
622 if base.IsETXTBSY(err) {
623
624
625
626 return nil
627 }
628 return err
629 }
630 defer f.Close()
631 if size == 0 {
632
633
634
635 return nil
636 }
637
638
639
640
641
642
643 if _, err := file.Seek(0, 0); err != nil {
644 f.Truncate(0)
645 return err
646 }
647 h := sha256.New()
648 w := io.MultiWriter(f, h)
649 if _, err := io.CopyN(w, file, size-1); err != nil {
650 f.Truncate(0)
651 return err
652 }
653
654
655
656 buf := make([]byte, 1)
657 if _, err := file.Read(buf); err != nil {
658 f.Truncate(0)
659 return err
660 }
661 h.Write(buf)
662 sum := h.Sum(nil)
663 if !bytes.Equal(sum, out[:]) {
664 f.Truncate(0)
665 return fmt.Errorf("file content changed underfoot")
666 }
667
668
669 if _, err := f.Write(buf); err != nil {
670 f.Truncate(0)
671 return err
672 }
673 if err := f.Close(); err != nil {
674
675
676
677 os.Remove(name)
678 return err
679 }
680 os.Chtimes(name, c.now(), c.now())
681
682 return nil
683 }
684
685
686
687
688
689
690
691
692
693 func (c *DiskCache) FuzzDir() string {
694 return filepath.Join(c.dir, "fuzz")
695 }
696
View as plain text