func (me *store) OpenTorrentData(info *metainfo.Info) (ret *data) { ret = &data{info, me} for i := range iter.N(info.NumPieces()) { go ret.PieceComplete(i) } return }
func extentCompleteRequiredLengths(info *metainfo.Info, off, n int64) (ret []metainfo.FileInfo) { if n == 0 { return } for _, fi := range info.UpvertedFiles() { if off >= fi.Length { off -= fi.Length continue } n1 := n if off+n1 > fi.Length { n1 = fi.Length - off } ret = append(ret, metainfo.FileInfo{ Path: fi.Path, Length: off + n1, }) n -= n1 if n == 0 { return } off = 0 } panic("extent exceeds torrent bounds") }
func TorrentData(md *metainfo.Info, location string) (ret *torrentData, err error) { var mms mmap_span.MMapSpan defer func() { if err != nil { mms.Close() } }() for _, miFile := range md.UpvertedFiles() { fileName := filepath.Join(append([]string{location, md.Name}, miFile.Path...)...) err = os.MkdirAll(filepath.Dir(fileName), 0777) if err != nil { err = fmt.Errorf("error creating data directory %q: %s", filepath.Dir(fileName), err) return } var file *os.File file, err = os.OpenFile(fileName, os.O_CREATE|os.O_RDWR, 0666) if err != nil { return } func() { defer file.Close() var fi os.FileInfo fi, err = file.Stat() if err != nil { return } if fi.Size() < miFile.Length { err = file.Truncate(miFile.Length) if err != nil { return } } if miFile.Length == 0 { // Can't mmap() regions with length 0. return } var mMap mmap.MMap mMap, err = mmap.MapRegion(file, int(miFile.Length), // Probably not great on <64 bit systems. mmap.RDWR, 0, 0) if err != nil { err = fmt.Errorf("error mapping file %q, length %d: %s", file.Name(), miFile.Length, err) return } if int64(len(mMap)) != miFile.Length { panic("mmap has wrong length") } mms.Append(mMap) }() if err != nil { return } } ret = &torrentData{ MMapSpan: mms, completed: make([]bool, md.NumPieces()), } return }
func validateInfo(info *metainfo.Info) error { if len(info.Pieces)%20 != 0 { return errors.New("pieces has invalid length") } if int((info.TotalLength()+info.PieceLength-1)/info.PieceLength) != info.NumPieces() { return errors.New("piece count and file lengths are at odds") } return nil }
func TorrentData(md *metainfo.Info, location string) (mms *mmap_span.MMapSpan, err error) { mms = &mmap_span.MMapSpan{} defer func() { if err != nil { mms.Close() mms = nil } }() for _, miFile := range md.UpvertedFiles() { fileName := filepath.Join(append([]string{location, md.Name}, miFile.Path...)...) err = os.MkdirAll(filepath.Dir(fileName), 0777) if err != nil { err = fmt.Errorf("error creating data directory %q: %s", filepath.Dir(fileName), err) return } var file *os.File file, err = os.OpenFile(fileName, os.O_CREATE|os.O_RDWR, 0666) if err != nil { return } func() { defer file.Close() var fi os.FileInfo fi, err = file.Stat() if err != nil { return } if fi.Size() < miFile.Length { err = file.Truncate(miFile.Length) if err != nil { return } } if miFile.Length == 0 { // Can't mmap() regions with length 0. return } var mMap gommap.MMap mMap, err = gommap.MapRegion(file.Fd(), 0, miFile.Length, gommap.PROT_READ|gommap.PROT_WRITE, gommap.MAP_SHARED) if err != nil { err = fmt.Errorf("error mapping file %q, length %d: %s", file.Name(), miFile.Length, err) return } if int64(len(mMap)) != miFile.Length { panic("mmap has wrong length") } mms.Append(mMap) }() if err != nil { return } } return }
func TorrentData(md *metainfo.Info, location string) data { return data{md, location, make([]bool, md.NumPieces())} }