Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
B
bpftrace
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
Kirill Smelkov
bpftrace
Commits
3fe276e2
Commit
3fe276e2
authored
Jan 04, 2019
by
Dan Xu
Committed by
GitHub
Jan 04, 2019
Browse files
Options
Browse Files
Download
Plain Diff
Merge pull request #301 from Birch-san/fix-get-arg-values-use-after-free
Fix use-after-free in BPFtrace::get_arg_values
parents
a349c28b
01491c63
Changes
4
Show whitespace changes
Inline
Side-by-side
Showing
4 changed files
with
126 additions
and
49 deletions
+126
-49
src/bpftrace.cpp
src/bpftrace.cpp
+72
-48
src/bpftrace.h
src/bpftrace.h
+2
-1
src/printf.cpp
src/printf.cpp
+15
-0
src/printf.h
src/printf.h
+37
-0
No files found.
src/bpftrace.cpp
View file @
3fe276e2
...
...
@@ -21,6 +21,7 @@
#include "bpforc.h"
#include "bpftrace.h"
#include "attached_probe.h"
#include "printf.h"
#include "triggers.h"
#include "resolve_cgroupid.h"
...
...
@@ -263,7 +264,7 @@ void perf_event_printer(void *cb_cookie, void *data, int size)
auto
id
=
printf_id
-
asyncactionint
(
AsyncAction
::
syscall
);
auto
fmt
=
std
::
get
<
0
>
(
bpftrace
->
system_args_
[
id
]).
c_str
();
auto
args
=
std
::
get
<
1
>
(
bpftrace
->
system_args_
[
id
]);
std
::
vector
<
uint64_t
>
arg_values
=
bpftrace
->
get_arg_values
(
args
,
arg_data
);
auto
arg_values
=
bpftrace
->
get_arg_values
(
args
,
arg_data
);
char
buffer
[
255
];
...
...
@@ -273,30 +274,30 @@ void perf_event_printer(void *cb_cookie, void *data, int size)
system
(
fmt
);
break
;
case
1
:
snprintf
(
buffer
,
255
,
fmt
,
arg_values
.
at
(
0
));
snprintf
(
buffer
,
255
,
fmt
,
arg_values
.
at
(
0
)
->
value
()
);
system
(
buffer
);
break
;
case
2
:
snprintf
(
buffer
,
255
,
fmt
,
arg_values
.
at
(
0
)
,
arg_values
.
at
(
1
));
snprintf
(
buffer
,
255
,
fmt
,
arg_values
.
at
(
0
)
->
value
(),
arg_values
.
at
(
1
)
->
value
(
));
system
(
buffer
);
break
;
case
3
:
snprintf
(
buffer
,
255
,
fmt
,
arg_values
.
at
(
0
)
,
arg_values
.
at
(
1
),
arg_values
.
at
(
2
));
snprintf
(
buffer
,
255
,
fmt
,
arg_values
.
at
(
0
)
->
value
(),
arg_values
.
at
(
1
)
->
value
(),
arg_values
.
at
(
2
)
->
value
(
));
system
(
buffer
);
break
;
case
4
:
snprintf
(
buffer
,
255
,
fmt
,
arg_values
.
at
(
0
)
,
arg_values
.
at
(
1
),
arg_values
.
at
(
2
),
arg_values
.
at
(
3
));
snprintf
(
buffer
,
255
,
fmt
,
arg_values
.
at
(
0
)
->
value
(),
arg_values
.
at
(
1
)
->
value
(),
arg_values
.
at
(
2
)
->
value
(
),
arg_values
.
at
(
3
)
->
value
()
);
system
(
buffer
);
break
;
case
5
:
snprintf
(
buffer
,
255
,
fmt
,
arg_values
.
at
(
0
)
,
arg_values
.
at
(
1
),
arg_values
.
at
(
2
),
arg_values
.
at
(
3
)
,
arg_values
.
at
(
4
));
snprintf
(
buffer
,
255
,
fmt
,
arg_values
.
at
(
0
)
->
value
(),
arg_values
.
at
(
1
)
->
value
(),
arg_values
.
at
(
2
)
->
value
(
),
arg_values
.
at
(
3
)
->
value
(),
arg_values
.
at
(
4
)
->
value
(
));
system
(
buffer
);
break
;
case
6
:
snprintf
(
buffer
,
255
,
fmt
,
arg_values
.
at
(
0
)
,
arg_values
.
at
(
1
),
arg_values
.
at
(
2
),
arg_values
.
at
(
3
)
,
arg_values
.
at
(
4
),
arg_values
.
at
(
5
));
snprintf
(
buffer
,
255
,
fmt
,
arg_values
.
at
(
0
)
->
value
(),
arg_values
.
at
(
1
)
->
value
(),
arg_values
.
at
(
2
)
->
value
(
),
arg_values
.
at
(
3
)
->
value
(),
arg_values
.
at
(
4
)
->
value
(),
arg_values
.
at
(
5
)
->
value
(
));
system
(
buffer
);
break
;
default:
...
...
@@ -309,7 +310,7 @@ void perf_event_printer(void *cb_cookie, void *data, int size)
// printf
auto
fmt
=
std
::
get
<
0
>
(
bpftrace
->
printf_args_
[
printf_id
]).
c_str
();
auto
args
=
std
::
get
<
1
>
(
bpftrace
->
printf_args_
[
printf_id
]);
std
::
vector
<
uint64_t
>
arg_values
=
bpftrace
->
get_arg_values
(
args
,
arg_data
);
auto
arg_values
=
bpftrace
->
get_arg_values
(
args
,
arg_data
);
switch
(
args
.
size
())
{
...
...
@@ -317,38 +318,35 @@ void perf_event_printer(void *cb_cookie, void *data, int size)
printf
(
fmt
);
break
;
case
1
:
printf
(
fmt
,
arg_values
.
at
(
0
));
printf
(
fmt
,
arg_values
.
at
(
0
)
->
value
()
);
break
;
case
2
:
printf
(
fmt
,
arg_values
.
at
(
0
)
,
arg_values
.
at
(
1
));
printf
(
fmt
,
arg_values
.
at
(
0
)
->
value
(),
arg_values
.
at
(
1
)
->
value
(
));
break
;
case
3
:
printf
(
fmt
,
arg_values
.
at
(
0
)
,
arg_values
.
at
(
1
),
arg_values
.
at
(
2
));
printf
(
fmt
,
arg_values
.
at
(
0
)
->
value
(),
arg_values
.
at
(
1
)
->
value
(),
arg_values
.
at
(
2
)
->
value
(
));
break
;
case
4
:
printf
(
fmt
,
arg_values
.
at
(
0
)
,
arg_values
.
at
(
1
),
arg_values
.
at
(
2
),
arg_values
.
at
(
3
));
printf
(
fmt
,
arg_values
.
at
(
0
)
->
value
(),
arg_values
.
at
(
1
)
->
value
(),
arg_values
.
at
(
2
)
->
value
(
),
arg_values
.
at
(
3
)
->
value
(
));
break
;
case
5
:
printf
(
fmt
,
arg_values
.
at
(
0
)
,
arg_values
.
at
(
1
),
arg_values
.
at
(
2
),
arg_values
.
at
(
3
),
arg_values
.
at
(
4
));
printf
(
fmt
,
arg_values
.
at
(
0
)
->
value
(),
arg_values
.
at
(
1
)
->
value
(),
arg_values
.
at
(
2
)
->
value
(
),
arg_values
.
at
(
3
)
->
value
(),
arg_values
.
at
(
4
)
->
value
(
));
break
;
case
6
:
printf
(
fmt
,
arg_values
.
at
(
0
)
,
arg_values
.
at
(
1
),
arg_values
.
at
(
2
),
arg_values
.
at
(
3
),
arg_values
.
at
(
4
),
arg_values
.
at
(
5
));
printf
(
fmt
,
arg_values
.
at
(
0
)
->
value
(),
arg_values
.
at
(
1
)
->
value
(),
arg_values
.
at
(
2
)
->
value
(
),
arg_values
.
at
(
3
)
->
value
(),
arg_values
.
at
(
4
)
->
value
(),
arg_values
.
at
(
5
)
->
value
(
));
break
;
default:
abort
();
}
}
std
::
vector
<
uint64_t
>
BPFtrace
::
get_arg_values
(
std
::
vector
<
Field
>
args
,
uint8_t
*
arg_data
)
std
::
vector
<
std
::
unique_ptr
<
IPrintable
>>
BPFtrace
::
get_arg_values
(
const
std
::
vector
<
Field
>
&
args
,
uint8_t
*
arg_data
)
{
std
::
vector
<
uint64_t
>
arg_values
;
std
::
vector
<
std
::
unique_ptr
<
char
>>
resolved_symbols
;
std
::
vector
<
std
::
unique_ptr
<
char
>>
resolved_usernames
;
std
::
vector
<
std
::
unique_ptr
<
IPrintable
>>
arg_values
;
char
*
name
;
for
(
auto
arg
:
args
)
{
switch
(
arg
.
type
.
type
)
...
...
@@ -357,54 +355,80 @@ std::vector<uint64_t> BPFtrace::get_arg_values(std::vector<Field> args, uint8_t*
switch
(
arg
.
type
.
size
)
{
case
8
:
arg_values
.
push_back
(
*
(
uint64_t
*
)(
arg_data
+
arg
.
offset
));
arg_values
.
push_back
(
std
::
make_unique
<
PrintableInt
>
(
*
reinterpret_cast
<
uint64_t
*>
(
arg_data
+
arg
.
offset
)));
break
;
case
4
:
arg_values
.
push_back
(
*
(
uint32_t
*
)(
arg_data
+
arg
.
offset
));
arg_values
.
push_back
(
std
::
make_unique
<
PrintableInt
>
(
*
reinterpret_cast
<
uint32_t
*>
(
arg_data
+
arg
.
offset
)));
break
;
case
2
:
arg_values
.
push_back
(
*
(
uint16_t
*
)(
arg_data
+
arg
.
offset
));
arg_values
.
push_back
(
std
::
make_unique
<
PrintableInt
>
(
*
reinterpret_cast
<
uint16_t
*>
(
arg_data
+
arg
.
offset
)));
break
;
case
1
:
arg_values
.
push_back
(
*
(
uint8_t
*
)(
arg_data
+
arg
.
offset
));
arg_values
.
push_back
(
std
::
make_unique
<
PrintableInt
>
(
*
reinterpret_cast
<
uint8_t
*>
(
arg_data
+
arg
.
offset
)));
break
;
default:
abort
();
}
break
;
case
Type
:
:
string
:
arg_values
.
push_back
((
uint64_t
)(
arg_data
+
arg
.
offset
));
arg_values
.
push_back
(
std
::
make_unique
<
PrintableCString
>
(
reinterpret_cast
<
char
*>
(
arg_data
+
arg
.
offset
)));
break
;
case
Type
:
:
sym
:
resolved_symbols
.
emplace_back
(
strdup
(
resolve_sym
(
*
(
uint64_t
*
)(
arg_data
+
arg
.
offset
)).
c_str
()));
arg_values
.
push_back
((
uint64_t
)
resolved_symbols
.
back
().
get
(
));
arg_values
.
push_back
(
std
::
make_unique
<
PrintableString
>
(
resolve_sym
(
*
reinterpret_cast
<
uint64_t
*>
(
arg_data
+
arg
.
offset
))
));
break
;
case
Type
:
:
usym
:
resolved_symbols
.
emplace_back
(
strdup
(
resolve_usym
(
*
(
uint64_t
*
)(
arg_data
+
arg
.
offset
),
*
(
uint64_t
*
)(
arg_data
+
arg
.
offset
+
8
)).
c_str
()));
arg_values
.
push_back
((
uint64_t
)
resolved_symbols
.
back
().
get
());
arg_values
.
push_back
(
std
::
make_unique
<
PrintableString
>
(
resolve_usym
(
*
reinterpret_cast
<
uint64_t
*>
(
arg_data
+
arg
.
offset
),
*
reinterpret_cast
<
uint64_t
*>
(
arg_data
+
arg
.
offset
+
8
))));
break
;
case
Type
:
:
inet
:
name
=
strdup
(
resolve_inet
(
*
(
uint64_t
*
)(
arg_data
+
arg
.
offset
),
*
(
uint64_t
*
)(
arg_data
+
arg
.
offset
+
8
)).
c_str
());
arg_values
.
push_back
((
uint64_t
)
name
);
arg_values
.
push_back
(
std
::
make_unique
<
PrintableString
>
(
resolve_inet
(
*
reinterpret_cast
<
uint64_t
*>
(
arg_data
+
arg
.
offset
),
*
reinterpret_cast
<
uint64_t
*>
(
arg_data
+
arg
.
offset
+
8
))));
break
;
case
Type
:
:
username
:
resolved_usernames
.
emplace_back
(
strdup
(
resolve_uid
(
*
(
uint64_t
*
)(
arg_data
+
arg
.
offset
)).
c_str
()));
arg_values
.
push_back
((
uint64_t
)
resolved_usernames
.
back
().
get
());
arg_values
.
push_back
(
std
::
make_unique
<
PrintableString
>
(
resolve_uid
(
*
reinterpret_cast
<
uint64_t
*>
(
arg_data
+
arg
.
offset
))));
break
;
case
Type
:
:
probe
:
name
=
strdup
(
resolve_probe
(
*
(
uint64_t
*
)(
arg_data
+
arg
.
offset
)).
c_str
());
arg_values
.
push_back
((
uint64_t
)
name
);
arg_values
.
push_back
(
std
::
make_unique
<
PrintableString
>
(
resolve_probe
(
*
reinterpret_cast
<
uint64_t
*>
(
arg_data
+
arg
.
offset
))));
break
;
case
Type
:
:
stack
:
name
=
strdup
(
get_stack
(
*
(
uint64_t
*
)(
arg_data
+
arg
.
offset
),
false
,
8
).
c_str
());
arg_values
.
push_back
((
uint64_t
)
name
);
arg_values
.
push_back
(
std
::
make_unique
<
PrintableString
>
(
get_stack
(
*
reinterpret_cast
<
uint64_t
*>
(
arg_data
+
arg
.
offset
),
false
,
8
)));
break
;
case
Type
:
:
ustack
:
name
=
strdup
(
get_stack
(
*
(
uint64_t
*
)(
arg_data
+
arg
.
offset
),
true
,
8
).
c_str
());
arg_values
.
push_back
((
uint64_t
)
name
);
arg_values
.
push_back
(
std
::
make_unique
<
PrintableString
>
(
get_stack
(
*
reinterpret_cast
<
uint64_t
*>
(
arg_data
+
arg
.
offset
),
true
,
8
)));
break
;
default:
abort
();
...
...
src/bpftrace.h
View file @
3fe276e2
...
...
@@ -11,6 +11,7 @@
#include "ast.h"
#include "attached_probe.h"
#include "imap.h"
#include "printf.h"
#include "struct.h"
#include "types.h"
...
...
@@ -68,7 +69,7 @@ public:
std
::
string
extract_func_symbols_from_path
(
const
std
::
string
&
path
);
std
::
string
resolve_probe
(
uint64_t
probe_id
);
uint64_t
resolve_cgroupid
(
const
std
::
string
&
path
);
std
::
vector
<
uint64_t
>
get_arg_values
(
std
::
vector
<
Field
>
args
,
uint8_t
*
arg_data
);
std
::
vector
<
std
::
unique_ptr
<
IPrintable
>>
get_arg_values
(
const
std
::
vector
<
Field
>
&
args
,
uint8_t
*
arg_data
);
void
add_param
(
const
std
::
string
&
param
);
bool
is_numeric
(
std
::
string
str
);
std
::
string
get_param
(
int
index
);
...
...
src/printf.cpp
View file @
3fe276e2
...
...
@@ -64,4 +64,19 @@ std::string verify_format_string(const std::string &fmt, std::vector<Field> args
return
""
;
}
uint64_t
PrintableString
::
value
()
{
return
(
uint64_t
)
value_
.
c_str
();
}
uint64_t
PrintableCString
::
value
()
{
return
(
uint64_t
)
value_
;
}
uint64_t
PrintableInt
::
value
()
{
return
value_
;
}
}
// namespace bpftrace
src/printf.h
View file @
3fe276e2
#pragma once
#include <sstream>
#include "ast.h"
...
...
@@ -9,4 +11,39 @@ struct Field;
std
::
string
verify_format_string
(
const
std
::
string
&
fmt
,
std
::
vector
<
Field
>
args
);
class
IPrintable
{
public:
virtual
~
IPrintable
()
{
};
virtual
uint64_t
value
()
=
0
;
};
class
PrintableString
:
public
virtual
IPrintable
{
public:
PrintableString
(
std
::
string
value
)
:
value_
(
std
::
move
(
value
))
{
}
uint64_t
value
();
private:
std
::
string
value_
;
};
class
PrintableCString
:
public
virtual
IPrintable
{
public:
PrintableCString
(
char
*
value
)
:
value_
(
value
)
{
}
uint64_t
value
();
private:
char
*
value_
;
};
class
PrintableInt
:
public
virtual
IPrintable
{
public:
PrintableInt
(
uint64_t
value
)
:
value_
(
value
)
{
}
uint64_t
value
();
private:
uint64_t
value_
;
};
}
// namespace bpftrace
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