Skip to content
Projects
Groups
Snippets
Help
Loading...
Sign in / Register
Toggle navigation
C
capnproto
Project
Project
Details
Activity
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Board
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Packages
Packages
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
submodule
capnproto
Commits
86ce2792
Commit
86ce2792
authored
Mar 12, 2017
by
Kenton Varda
Committed by
GitHub
Mar 12, 2017
Browse files
Options
Browse Files
Download
Plain Diff
Merge pull request #423 from dwrensha/canonical
fix various bugs in canonicalization
parents
8ceb4ce1
be4ac4e6
Hide whitespace changes
Inline
Side-by-side
Showing
4 changed files
with
139 additions
and
12 deletions
+139
-12
canonicalize-test.c++
c++/src/capnp/canonicalize-test.c++
+120
-2
layout.c++
c++/src/capnp/layout.c++
+12
-6
layout.h
c++/src/capnp/layout.h
+1
-1
message.c++
c++/src/capnp/message.c++
+6
-3
No files found.
c++/src/capnp/canonicalize-test.c++
View file @
86ce2792
...
...
@@ -37,8 +37,14 @@ KJ_TEST("canonicalize yields canonical message") {
initTestMessage
(
root
);
canonicalize
(
root
.
asReader
());
//Will assert if canonicalize failed to do so
auto
canonicalWords
=
canonicalize
(
root
.
asReader
());
// Throws an exception on canonicalization failure.
kj
::
ArrayPtr
<
const
capnp
::
word
>
canonicalSegments
[
1
]
=
{
canonicalWords
.
asPtr
()};
capnp
::
SegmentArrayMessageReader
canonicalReader
(
kj
::
arrayPtr
(
canonicalSegments
,
1
));
KJ_ASSERT
(
AnyStruct
::
Reader
(
root
.
asReader
())
==
AnyStruct
::
Reader
(
canonicalReader
.
getRoot
<
TestAllTypes
>
()));
}
KJ_TEST
(
"canonicalize succeeds on empty struct"
)
{
...
...
@@ -48,6 +54,56 @@ KJ_TEST("canonicalize succeeds on empty struct") {
canonicalize
(
root
.
asReader
());
// Throws an exception on canoncalization failure.
}
KJ_TEST
(
"data word with only its most significant bit set does not get truncated"
)
{
AlignedData
<
3
>
segment
=
{{
// Struct pointer, body immediately follows, two data words
0x00
,
0x00
,
0x00
,
0x00
,
0x02
,
0x00
,
0x00
,
0x00
,
// First data word
0x11
,
0x11
,
0x11
,
0x11
,
0x11
,
0x11
,
0x11
,
0x11
,
// Second data word, all zero except most significant bit
0x00
,
0x00
,
0x00
,
0x00
,
0x00
,
0x00
,
0x00
,
0x80
,
}};
kj
::
ArrayPtr
<
const
word
>
segments
[
1
]
=
{
kj
::
arrayPtr
(
segment
.
words
,
3
)};
SegmentArrayMessageReader
messageReader
(
kj
::
arrayPtr
(
segments
,
1
));
KJ_ASSERT
(
messageReader
.
isCanonical
());
auto
canonicalWords
=
canonicalize
(
messageReader
.
getRoot
<
TestAllTypes
>
());
// At one point this failed because an off-by-one bug in canonicalization
// caused the second word of the data section to be truncated.
ASSERT_EQ
(
canonicalWords
.
asBytes
(),
kj
::
arrayPtr
(
segment
.
bytes
,
3
*
8
));
}
KJ_TEST
(
"INLINE_COMPOSITE data word with only its most significant bit set does not get truncated"
)
{
AlignedData
<
5
>
segment
=
{{
// Struct pointer, body immediately follows, one pointer
0x00
,
0x00
,
0x00
,
0x00
,
0x00
,
0x00
,
0x01
,
0x00
,
// List pointer, no offset, inline composite, two words long
0x01
,
0x00
,
0x00
,
0x00
,
0x17
,
0x00
,
0x00
,
0x00
,
// Tag word, list has one element with two data words and no pointers
0x04
,
0x00
,
0x00
,
0x00
,
0x02
,
0x00
,
0x00
,
0x00
,
// First data word
0x22
,
0x22
,
0x22
,
0x22
,
0x22
,
0x22
,
0x22
,
0x22
,
// Second data word, all zero except most significant bit
0x00
,
0x00
,
0x00
,
0x00
,
0x00
,
0x00
,
0x00
,
0x80
,
}};
kj
::
ArrayPtr
<
const
word
>
segments
[
1
]
=
{
kj
::
arrayPtr
(
segment
.
words
,
5
)};
SegmentArrayMessageReader
messageReader
(
kj
::
arrayPtr
(
segments
,
1
));
KJ_ASSERT
(
messageReader
.
isCanonical
());
auto
canonicalWords
=
canonicalize
(
messageReader
.
getRoot
<
TestLists
>
());
// At one point this failed because an off-by-one bug in canonicalization
// caused the second word of the data section to be truncated.
ASSERT_EQ
(
canonicalWords
.
asBytes
(),
kj
::
arrayPtr
(
segment
.
bytes
,
5
*
8
));
}
KJ_TEST
(
"canonical non-null empty struct field"
)
{
AlignedData
<
4
>
nonNullEmptyStruct
=
{{
// Struct pointer, body immediately follows, two pointer fields, no data.
...
...
@@ -168,6 +224,68 @@ KJ_TEST("isCanonical requires truncation of 0-valued struct fields") {
KJ_ASSERT
(
!
nonTruncated
.
isCanonical
());
}
KJ_TEST
(
"isCanonical rejects unused trailing words"
)
{
AlignedData
<
3
>
segment
=
{{
// Struct pointer, data in next word
0x00
,
0x00
,
0x00
,
0x00
,
0x01
,
0x00
,
0x00
,
0x00
,
// Data section of struct
0x00
,
0x01
,
0x02
,
0x03
,
0x04
,
0x05
,
0x06
,
0x07
,
// Trailing zero word
0x00
,
0x00
,
0x00
,
0x00
,
0x00
,
0x00
,
0x00
,
0x00
,
}};
kj
::
ArrayPtr
<
const
word
>
segments
[
1
]
=
{
kj
::
arrayPtr
(
segment
.
words
,
3
)};
SegmentArrayMessageReader
message
(
kj
::
arrayPtr
(
segments
,
1
));
KJ_ASSERT
(
!
message
.
isCanonical
());
}
KJ_TEST
(
"isCanonical accepts empty inline composite list of zero-sized structs"
)
{
AlignedData
<
3
>
segment
=
{{
// Struct pointer, pointer in next word
0x00
,
0x00
,
0x00
,
0x00
,
0x00
,
0x00
,
0x01
,
0x00
,
// List pointer, inline composite, zero words long
0x01
,
0x00
,
0x00
,
0x00
,
0x07
,
0x00
,
0x00
,
0x00
,
// Tag word
0x00
,
0x00
,
0x00
,
0x00
,
0x00
,
0x00
,
0x00
,
0x00
,
}};
kj
::
ArrayPtr
<
const
word
>
segments
[
1
]
=
{
kj
::
arrayPtr
(
segment
.
words
,
3
)};
SegmentArrayMessageReader
message
(
kj
::
arrayPtr
(
segments
,
1
));
KJ_ASSERT
(
message
.
isCanonical
());
}
KJ_TEST
(
"isCanonical rejects inline composite list with inaccurate word-length"
)
{
AlignedData
<
6
>
segment
=
{{
// Struct pointer, no offset, pointer section has two entries
0x00
,
0x00
,
0x00
,
0x00
,
0x00
,
0x00
,
0x02
,
0x00
,
// List pointer, offset of one, inline composite, two words long
// (The list only needs to be one word long to hold its actual elements;
// therefore this message is not canonical.)
0x05
,
0x00
,
0x00
,
0x00
,
0x17
,
0x00
,
0x00
,
0x00
,
// Struct pointer, offset two, data section has one word
0x08
,
0x00
,
0x00
,
0x00
,
0x01
,
0x00
,
0x00
,
0x00
,
// Tag word, struct, one element, one word data section
0x04
,
0x00
,
0x00
,
0x00
,
0x01
,
0x00
,
0x00
,
0x00
,
// Data section of struct element of list
0x00
,
0x01
,
0x02
,
0x03
,
0x04
,
0x05
,
0x06
,
0x07
,
// Data section of struct field in top-level struct
0x07
,
0x06
,
0x05
,
0x04
,
0x03
,
0x02
,
0x01
,
0x00
,
}};
kj
::
ArrayPtr
<
const
word
>
segments
[
1
]
=
{
kj
::
arrayPtr
(
segment
.
words
,
6
)};
SegmentArrayMessageReader
message
(
kj
::
arrayPtr
(
segments
,
1
));
KJ_ASSERT
(
!
message
.
isCanonical
());
}
KJ_TEST
(
"upgraded lists can be canonicalized"
)
{
AlignedData
<
7
>
upgradedList
=
{{
//Struct pointer, data immediately follows, 4 pointer fields, no data
...
...
c++/src/capnp/layout.c++
View file @
86ce2792
...
...
@@ -1571,12 +1571,12 @@ struct WireHelpers {
}
else
{
// Truncate the data section
while
(
dataSize
!=
0
*
BYTES
)
{
size_t
end
=
(
dataSize
-
1
*
BYTES
)
/
BYTES
;
size_t
end
=
dataSize
/
BYTES
;
ByteCount
window
=
dataSize
%
BYTES_PER_WORD
;
if
(
window
==
0
*
BYTES
)
{
window
=
BYTES_PER_WORD
*
WORDS
;
}
size_t
start
=
end
+
1
-
window
/
BYTES
;
size_t
start
=
end
-
window
/
BYTES
;
kj
::
ArrayPtr
<
const
byte
>
lastWord
=
value
.
getDataSectionAsBlob
().
slice
(
start
,
end
);
bool
lastWordZero
=
true
;
//TODO(MRM) once this is known to work, replace with fast memcmp
...
...
@@ -1683,9 +1683,9 @@ struct WireHelpers {
auto
se
=
value
.
getStructElement
(
ec
);
WordCount
localDataSize
=
declDataSize
;
while
(
localDataSize
!=
0
*
WORDS
)
{
size_t
end
=
(
localDataSize
*
BYTES_PER_WORD
-
1
*
BYTES
)
/
BYTES
;
size_t
end
=
(
localDataSize
*
BYTES_PER_WORD
*
BYTES
)
/
BYTES
;
ByteCount
window
=
BYTES_PER_WORD
*
WORDS
;
size_t
start
=
end
+
1
-
window
/
BYTES
;
size_t
start
=
end
-
window
/
BYTES
;
kj
::
ArrayPtr
<
const
byte
>
lastWord
=
se
.
getDataSectionAsBlob
().
slice
(
start
,
end
);
bool
lastWordZero
=
true
;
//TODO(MRM) once this is known to work, replace with fast memcmp
...
...
@@ -2609,7 +2609,7 @@ bool PointerReader::isCanonical(const word **readHead) {
}
}
case
PointerType
:
:
LIST
:
return
this
->
getListAnySize
(
nullptr
).
isCanonical
(
readHead
);
return
this
->
getListAnySize
(
nullptr
).
isCanonical
(
readHead
,
pointer
);
case
PointerType
:
:
CAPABILITY
:
KJ_FAIL_ASSERT
(
"Capabilities are not positional"
);
}
...
...
@@ -2952,7 +2952,7 @@ ListReader ListReader::imbue(CapTableReader* capTable) const {
return
result
;
}
bool
ListReader
::
isCanonical
(
const
word
**
readHead
)
{
bool
ListReader
::
isCanonical
(
const
word
**
readHead
,
const
WirePointer
*
ref
)
{
switch
(
this
->
getElementSize
())
{
case
ElementSize
:
:
INLINE_COMPOSITE
:
{
*
readHead
+=
1
;
...
...
@@ -2966,6 +2966,12 @@ bool ListReader::isCanonical(const word **readHead) {
}
auto
structSize
=
(
this
->
structDataSize
/
BITS_PER_WORD
)
+
(
this
->
structPointerCount
*
WORDS_PER_POINTER
);
if
(
structSize
*
this
->
elementCount
!=
ref
->
listRef
.
inlineCompositeWordCount
())
{
return
false
;
}
if
(
structSize
==
0
*
WORDS
)
{
return
true
;
}
auto
listEnd
=
*
readHead
+
(
this
->
elementCount
/
ELEMENTS
*
structSize
)
/
WORDS
;
auto
pointerHead
=
listEnd
;
bool
listDataTrunc
=
false
;
...
...
c++/src/capnp/layout.h
View file @
86ce2792
...
...
@@ -769,7 +769,7 @@ public:
ListReader
imbue
(
CapTableReader
*
capTable
)
const
;
// Return a copy of this reader except using the given capability context.
bool
isCanonical
(
const
word
**
readHead
);
bool
isCanonical
(
const
word
**
readHead
,
const
WirePointer
*
ref
);
// Validate this pointer's canonicity, subject to the conditions:
// * All data to the left of readHead has been read thus far (for pointer
// ordering)
...
...
c++/src/capnp/message.c++
View file @
86ce2792
...
...
@@ -75,9 +75,12 @@ bool MessageReader::isCanonical() {
}
const
word
*
readHead
=
segment
->
getStartPtr
()
+
1
;
return
_
::
PointerReader
::
getRoot
(
segment
,
nullptr
,
segment
->
getStartPtr
(),
this
->
getOptions
().
nestingLimit
)
.
isCanonical
(
&
readHead
);
bool
rootIsCanonical
=
_
::
PointerReader
::
getRoot
(
segment
,
nullptr
,
segment
->
getStartPtr
(),
this
->
getOptions
().
nestingLimit
)
.
isCanonical
(
&
readHead
);
bool
allWordsConsumed
=
segment
->
getOffsetTo
(
readHead
)
==
segment
->
getSize
();
return
rootIsCanonical
&&
allWordsConsumed
;
}
...
...
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