9 "github.com/anacrolix/missinggo"
10 "github.com/edsrzf/mmap-go"
12 "github.com/anacrolix/torrent/metainfo"
13 "github.com/anacrolix/torrent/mmap_span"
16 type mmapStorage struct {
18 spans map[metainfo.InfoHash]mmap_span.MMapSpan
19 completed map[metainfo.InfoHash]bool
22 func NewMMap(baseDir string) I {
28 func (me *mmapStorage) lazySpan(info *metainfo.InfoEx) error {
30 me.spans = make(map[metainfo.InfoHash]mmap_span.MMapSpan)
32 if _, ok := me.spans[*info.Hash]; ok {
35 span, err := MMapTorrent(&info.Info, me.baseDir)
39 me.spans[*info.Hash] = span
43 func (me *mmapStorage) Piece(p metainfo.Piece) Piece {
44 err := me.lazySpan(p.Info)
48 return mmapStoragePiece{
51 ReaderAt: io.NewSectionReader(me.spans[*p.Info.Hash], p.Offset(), p.Length()),
52 WriterAt: missinggo.NewSectionWriter(me.spans[*p.Info.Hash], p.Offset(), p.Length()),
56 type mmapStoragePiece struct {
63 func (me mmapStoragePiece) GetIsComplete() bool {
64 return me.storage.completed[me.p.Hash()]
67 func (me mmapStoragePiece) MarkComplete() error {
68 if me.storage.completed == nil {
69 me.storage.completed = make(map[metainfo.InfoHash]bool)
71 me.storage.completed[me.p.Hash()] = true
75 func MMapTorrent(md *metainfo.Info, location string) (mms mmap_span.MMapSpan, err error) {
81 for _, miFile := range md.UpvertedFiles() {
82 fileName := filepath.Join(append([]string{location, md.Name}, miFile.Path...)...)
83 err = os.MkdirAll(filepath.Dir(fileName), 0777)
85 err = fmt.Errorf("error creating data directory %q: %s", filepath.Dir(fileName), err)
89 file, err = os.OpenFile(fileName, os.O_CREATE|os.O_RDWR, 0666)
100 if fi.Size() < miFile.Length {
101 err = file.Truncate(miFile.Length)
106 if miFile.Length == 0 {
107 // Can't mmap() regions with length 0.
111 mMap, err = mmap.MapRegion(file,
112 int(miFile.Length), // Probably not great on <64 bit systems.
115 err = fmt.Errorf("error mapping file %q, length %d: %s", file.Name(), miFile.Length, err)
118 if int64(len(mMap)) != miFile.Length {
119 panic("mmap has wrong length")