Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
W
wendelin.core
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
Joshua
wendelin.core
Commits
a634a113
Commit
a634a113
authored
Feb 21, 2020
by
Kirill Smelkov
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
.
parent
3c206aac
Changes
2
Hide whitespace changes
Inline
Side-by-side
Showing
2 changed files
with
37 additions
and
14 deletions
+37
-14
wcfs/client/wcfs.cpp
wcfs/client/wcfs.cpp
+35
-14
wcfs/client/wcfs.h
wcfs/client/wcfs.h
+2
-0
No files found.
wcfs/client/wcfs.cpp
View file @
a634a113
...
@@ -466,7 +466,7 @@ error _Conn::resync(zodb::Tid at) {
...
@@ -466,7 +466,7 @@ error _Conn::resync(zodb::Tid at) {
return
err
;
// XXX -> wconn down on err ?
return
err
;
// XXX -> wconn down on err ?
//
write-lock wconn._atMu
. This excludes everything else, and in
//
lock wconn._atMu.W
. This excludes everything else, and in
// particular _pinner_, from running and mutating files and mappings.
// particular _pinner_, from running and mutating files and mappings.
//
//
// NOTE we'll relock atMu as R in the second part of resync, so we prelock
// NOTE we'll relock atMu as R in the second part of resync, so we prelock
...
@@ -530,6 +530,8 @@ error _Conn::resync(zodb::Tid at) {
...
@@ -530,6 +530,8 @@ error _Conn::resync(zodb::Tid at) {
if
(
!
(
headfsize
%
f
->
blksize
==
0
))
if
(
!
(
headfsize
%
f
->
blksize
==
0
))
return
E
(
fmt
::
errorf
(
"wcfs bug: head/file size %% blksize != 0"
));
return
E
(
fmt
::
errorf
(
"wcfs bug: head/file size %% blksize != 0"
));
// replace zero regions in f mappings in accordance to adjusted f._headfsize.
// NOTE it is ok to access f._mmaps without locking f._mu because we hold wconn.atMu.W
for
(
auto
mmap
:
f
->
_mmaps
)
{
for
(
auto
mmap
:
f
->
_mmaps
)
{
//printf(" resync -> %s: unzero [%lu:%lu)", v(at), f->_headfsize/f->blksize, headfsize/f->blksize);
//printf(" resync -> %s: unzero [%lu:%lu)", v(at), f->_headfsize/f->blksize, headfsize/f->blksize);
uint8_t
*
mem_unzero_start
=
min
(
mmap
->
mem_stop
,
uint8_t
*
mem_unzero_start
=
min
(
mmap
->
mem_stop
,
...
@@ -571,6 +573,7 @@ error _Conn::resync(zodb::Tid at) {
...
@@ -571,6 +573,7 @@ error _Conn::resync(zodb::Tid at) {
zodb
::
Oid
foid
=
fit
.
first
;
zodb
::
Oid
foid
=
fit
.
first
;
//FileH f = fit.second;
//FileH f = fit.second;
// XXX need to lock f.mu because wconn.atMu is only R now.
string
ack
;
string
ack
;
tie
(
ack
,
err
)
=
wconn
.
_wlink
->
sendReq
(
context
::
background
(),
fmt
::
sprintf
(
"watch %s @%s"
,
v
(
foid
),
v
(
at
)));
tie
(
ack
,
err
)
=
wconn
.
_wlink
->
sendReq
(
context
::
background
(),
fmt
::
sprintf
(
"watch %s @%s"
,
v
(
foid
),
v
(
at
)));
if
(
err
!=
nil
)
if
(
err
!=
nil
)
...
@@ -702,37 +705,54 @@ error _FileH::close() {
...
@@ -702,37 +705,54 @@ error _FileH::close() {
_FileH
&
fileh
=
*
this
;
_FileH
&
fileh
=
*
this
;
Conn
wconn
=
fileh
.
wconn
;
Conn
wconn
=
fileh
.
wconn
;
// XXX decref open count
// XXX fileh.close can be called several times and after first call another
// XXX fileh could be opened for the same foid. Be careful not to interfere with it.
// -> just return nil for second close
// XXX locking ok?
// XXX locking ok?
wconn
->
_atMu
.
RLock
();
wconn
->
_atMu
.
RLock
();
fileh
.
_mu
.
lock
();
defer
([
&
]()
{
defer
([
&
]()
{
fileh
.
_mu
.
unlock
();
wconn
->
_atMu
.
RUnlock
();
wconn
->
_atMu
.
RUnlock
();
});
});
// fileh.close can be called several times. just return nil for second close.
if
(
fileh
.
_closed
)
return
nil
;
// decref open count; do real close only when last open goes away.
if
(
fileh
.
_nopen
<=
0
)
panic
(
"BUG: fileh._nopen <= 0"
);
fileh
.
_nopen
--
;
if
(
fileh
.
_nopen
>
0
)
return
nil
;
// last open went away - real close.
xerr
::
Contextf
E
(
"%s: close f<%s>"
,
v
(
wconn
),
v
(
fileh
.
foid
));
xerr
::
Contextf
E
(
"%s: close f<%s>"
,
v
(
wconn
),
v
(
fileh
.
foid
));
error
err
,
eret
;
auto
reterr1
=
[
&
eret
](
error
err
)
{
if
(
eret
==
nil
&&
err
!=
nil
)
eret
=
err
;
};
// XXX change all fileh.mmaps to cause EFAULT on any access after fileh.close
// XXX change all fileh.mmaps to cause EFAULT on any access after fileh.close
// stop watching f
// stop watching f
XXX ok under f.mu ?
string
ack
;
string
ack
;
error
err
;
tie
(
ack
,
err
)
=
wconn
->
_wlink
->
sendReq
(
context
::
background
(),
fmt
::
sprintf
(
"watch %s -"
,
v
(
foid
)));
tie
(
ack
,
err
)
=
wconn
->
_wlink
->
sendReq
(
context
::
background
(),
fmt
::
sprintf
(
"watch %s -"
,
v
(
foid
)));
if
(
err
!=
nil
)
if
(
err
!=
nil
)
ret
urn
E
(
err
);
ret
err1
(
err
);
if
(
ack
!=
"ok"
)
else
if
(
ack
!=
"ok"
)
ret
urn
E
(
fmt
::
errorf
(
"unwatch: %s"
,
v
(
ack
)));
ret
err1
(
fmt
::
errorf
(
"unwatch: %s"
,
v
(
ack
)));
// remove fileh from wconn._filehTab
// remove fileh from wconn._filehTab
wconn
->
_mu
.
Lock
();
wconn
->
_mu
.
Lock
();
// FIXME lock order vs fileh._mu
if
(
wconn
->
_filehTab
.
get
(
fileh
.
foid
).
_ptr
()
==
&
fileh
)
// XXX ->
assert ?
if
(
wconn
->
_filehTab
.
get
(
fileh
.
foid
).
_ptr
()
==
&
fileh
)
// XXX ->
panic(BUG)
wconn
->
_filehTab
.
erase
(
fileh
.
foid
);
wconn
->
_filehTab
.
erase
(
fileh
.
foid
);
wconn
->
_mu
.
Unlock
();
wconn
->
_mu
.
Unlock
();
return
E
(
fileh
.
_headf
->
close
());
reterr1
(
fileh
.
_headf
->
close
());
fileh
.
_closed
=
true
;
return
E
(
eret
);
}
}
// mmap creates file mapping representing file[blk_start +blk_len) data as of wconn.at database state.
// mmap creates file mapping representing file[blk_start +blk_len) data as of wconn.at database state.
...
@@ -863,6 +883,7 @@ error _Mapping::unmap() {
...
@@ -863,6 +883,7 @@ error _Mapping::unmap() {
// XXX clear other fields?
// XXX clear other fields?
// XXX do it first? (to avoid pinner going through f.mmaps and hitting unmapped memory)
// XXX do it first? (to avoid pinner going through f.mmaps and hitting unmapped memory)
// -> no need: both pinner and unmap lock on f.mu
//f->_mmaps.remove(mmap);
//f->_mmaps.remove(mmap);
f
->
_mmaps
.
erase
(
f
->
_mmaps
.
erase
(
std
::
remove
(
f
->
_mmaps
.
begin
(),
f
->
_mmaps
.
end
(),
mmap
),
std
::
remove
(
f
->
_mmaps
.
begin
(),
f
->
_mmaps
.
end
(),
mmap
),
...
...
wcfs/client/wcfs.h
View file @
a634a113
...
@@ -238,6 +238,8 @@ struct _FileH : object {
...
@@ -238,6 +238,8 @@ struct _FileH : object {
sync
::
Mutex
_mu
;
// atMu.W | atMu.R + _mu
sync
::
Mutex
_mu
;
// atMu.W | atMu.R + _mu
dict
<
int64_t
,
zodb
::
Tid
>
_pinned
;
// {} blk -> rev that wcfs already sent us for this file
dict
<
int64_t
,
zodb
::
Tid
>
_pinned
;
// {} blk -> rev that wcfs already sent us for this file
vector
<
Mapping
>
_mmaps
;
// []Mapping ↑blk_start mappings of this file
vector
<
Mapping
>
_mmaps
;
// []Mapping ↑blk_start mappings of this file
int
_nopen
;
// number of times Conn.open returned this fileh
bool
_closed
;
// y after .close()
// don't new - create via Conn.open
// don't new - create via Conn.open
private:
private:
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment