2014-07-13 00:45:33 +02:00
|
|
|
// Copyright (C) 2014 Jakob Borg and Contributors (see the CONTRIBUTORS file).
|
2014-09-29 21:43:32 +02:00
|
|
|
//
|
|
|
|
// This program is free software: you can redistribute it and/or modify it
|
|
|
|
// under the terms of the GNU General Public License as published by the Free
|
|
|
|
// Software Foundation, either version 3 of the License, or (at your option)
|
|
|
|
// any later version.
|
|
|
|
//
|
|
|
|
// This program is distributed in the hope that it will be useful, but WITHOUT
|
|
|
|
// ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
|
|
// FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
|
|
|
// more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU General Public License along
|
|
|
|
// with this program. If not, see <http://www.gnu.org/licenses/>.
|
2014-06-01 22:50:14 +02:00
|
|
|
|
2014-05-15 00:26:55 -03:00
|
|
|
package model
|
2014-03-28 14:36:57 +01:00
|
|
|
|
2014-09-22 21:42:11 +02:00
|
|
|
import "github.com/syncthing/syncthing/internal/protocol"
|
2014-03-28 14:36:57 +01:00
|
|
|
|
|
|
|
type bqAdd struct {
|
2014-07-12 23:06:48 +02:00
|
|
|
file protocol.FileInfo
|
|
|
|
have []protocol.BlockInfo
|
|
|
|
need []protocol.BlockInfo
|
2014-03-28 14:36:57 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
type bqBlock struct {
|
2014-07-12 23:06:48 +02:00
|
|
|
file protocol.FileInfo
|
|
|
|
block protocol.BlockInfo // get this block from the network
|
|
|
|
copy []protocol.BlockInfo // copy these blocks from the old version of the file
|
2014-05-28 11:45:45 +02:00
|
|
|
first bool
|
2014-03-28 14:36:57 +01:00
|
|
|
last bool
|
|
|
|
}
|
|
|
|
|
|
|
|
type blockQueue struct {
|
|
|
|
queued []bqBlock
|
|
|
|
}
|
|
|
|
|
2014-07-24 09:38:16 +02:00
|
|
|
func (q *blockQueue) put(a bqAdd) {
|
2014-03-28 14:36:57 +01:00
|
|
|
// If we already have it queued, return
|
|
|
|
for _, b := range q.queued {
|
|
|
|
if b.file.Name == a.file.Name {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
2014-05-28 11:45:45 +02:00
|
|
|
|
|
|
|
l := len(a.need)
|
|
|
|
|
2014-03-28 14:36:57 +01:00
|
|
|
if len(a.have) > 0 {
|
|
|
|
// First queue a copy operation
|
|
|
|
q.queued = append(q.queued, bqBlock{
|
2014-05-28 11:45:45 +02:00
|
|
|
file: a.file,
|
|
|
|
copy: a.have,
|
|
|
|
first: true,
|
|
|
|
last: l == 0,
|
2014-03-28 14:36:57 +01:00
|
|
|
})
|
|
|
|
}
|
2014-05-28 11:45:45 +02:00
|
|
|
|
2014-03-28 14:36:57 +01:00
|
|
|
// Queue the needed blocks individually
|
|
|
|
for i, b := range a.need {
|
|
|
|
q.queued = append(q.queued, bqBlock{
|
|
|
|
file: a.file,
|
|
|
|
block: b,
|
2014-05-28 11:45:45 +02:00
|
|
|
first: len(a.have) == 0 && i == 0,
|
2014-03-28 14:36:57 +01:00
|
|
|
last: i == l-1,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2014-05-28 11:45:45 +02:00
|
|
|
if len(a.need)+len(a.have) == 0 {
|
2014-03-28 14:36:57 +01:00
|
|
|
// If we didn't have anything to fetch, queue an empty block with the "last" flag set to close the file.
|
|
|
|
q.queued = append(q.queued, bqBlock{
|
|
|
|
file: a.file,
|
|
|
|
last: true,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-07-24 09:38:16 +02:00
|
|
|
func (q *blockQueue) get() (bqBlock, bool) {
|
|
|
|
if len(q.queued) == 0 {
|
|
|
|
return bqBlock{}, false
|
2014-03-28 14:36:57 +01:00
|
|
|
}
|
2014-07-24 09:38:16 +02:00
|
|
|
b := q.queued[0]
|
|
|
|
q.queued = q.queued[1:]
|
|
|
|
return b, true
|
2014-03-28 14:36:57 +01:00
|
|
|
}
|