Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
B
bcc
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
bcc
Commits
f197faf4
Commit
f197faf4
authored
Jun 01, 2015
by
4ast
Browse files
Options
Browse Files
Download
Plain Diff
Merge pull request #24 from plumgrid/bblanco_dev
Add support for packet field write
parents
96783939
22a419b0
Changes
13
Show whitespace changes
Inline
Side-by-side
Showing
13 changed files
with
158 additions
and
45 deletions
+158
-45
src/bpf.py
src/bpf.py
+1
-1
src/cc/b_frontend_action.cc
src/cc/b_frontend_action.cc
+33
-3
src/cc/b_frontend_action.h
src/cc/b_frontend_action.h
+2
-1
src/cc/bpf_helpers.h
src/cc/bpf_helpers.h
+5
-0
src/cc/proto.h
src/cc/proto.h
+30
-30
tests/jit/CMakeLists.txt
tests/jit/CMakeLists.txt
+4
-2
tests/jit/test1.b
tests/jit/test1.b
+1
-1
tests/jit/test1.c
tests/jit/test1.c
+2
-2
tests/jit/test1.py
tests/jit/test1.py
+2
-2
tests/jit/test2.b
tests/jit/test2.b
+1
-1
tests/jit/test2.c
tests/jit/test2.c
+74
-0
tests/jit/test2.py
tests/jit/test2.py
+2
-2
tests/wrapper.sh.in
tests/wrapper.sh.in
+1
-0
No files found.
src/bpf.py
View file @
f197faf4
...
@@ -149,7 +149,7 @@ class BPF(object):
...
@@ -149,7 +149,7 @@ class BPF(object):
return
BPF
.
Table
(
self
,
map_fd
,
keytype
,
leaftype
)
return
BPF
.
Table
(
self
,
map_fd
,
keytype
,
leaftype
)
@
staticmethod
@
staticmethod
def
attach_socket
(
fn
,
dev
):
def
attach_
raw_
socket
(
fn
,
dev
):
if
not
isinstance
(
fn
,
BPF
.
Function
):
if
not
isinstance
(
fn
,
BPF
.
Function
):
raise
Exception
(
"arg 1 must be of type BPF.Function"
)
raise
Exception
(
"arg 1 must be of type BPF.Function"
)
sock
=
lib
.
bpf_open_raw_sock
(
dev
.
encode
(
"ascii"
))
sock
=
lib
.
bpf_open_raw_sock
(
dev
.
encode
(
"ascii"
))
...
...
src/cc/b_frontend_action.cc
View file @
f197faf4
...
@@ -148,12 +148,42 @@ bool BTypeVisitor::VisitArraySubscriptExpr(ArraySubscriptExpr *Arr) {
...
@@ -148,12 +148,42 @@ bool BTypeVisitor::VisitArraySubscriptExpr(ArraySubscriptExpr *Arr) {
return
true
;
return
true
;
}
}
bool
BTypeVisitor
::
VisitMemberExpr
(
MemberExpr
*
E
)
{
bool
BTypeVisitor
::
VisitBinaryOperator
(
BinaryOperator
*
E
)
{
Expr
*
Base
=
E
->
getBase
()
->
IgnoreImplicit
();
if
(
!
E
->
isAssignmentOp
())
return
true
;
Expr
*
LHS
=
E
->
getLHS
()
->
IgnoreImplicit
();
Expr
*
RHS
=
E
->
getRHS
()
->
IgnoreImplicit
();
if
(
MemberExpr
*
Memb
=
dyn_cast
<
MemberExpr
>
(
LHS
))
{
if
(
DeclRefExpr
*
Base
=
dyn_cast
<
DeclRefExpr
>
(
Memb
->
getBase
()
->
IgnoreImplicit
()))
{
if
(
DeprecatedAttr
*
A
=
Base
->
getDecl
()
->
getAttr
<
DeprecatedAttr
>
())
{
if
(
A
->
getMessage
()
==
"packet"
)
{
if
(
FieldDecl
*
F
=
dyn_cast
<
FieldDecl
>
(
Memb
->
getMemberDecl
()))
{
uint64_t
ofs
=
C
.
getFieldOffset
(
F
);
uint64_t
sz
=
F
->
isBitField
()
?
F
->
getBitWidthValue
(
C
)
:
C
.
getTypeSize
(
F
->
getType
());
string
base
=
rewriter_
.
getRewrittenText
(
SourceRange
(
Base
->
getLocStart
(),
Base
->
getLocEnd
()));
string
rhs
=
rewriter_
.
getRewrittenText
(
SourceRange
(
RHS
->
getLocStart
(),
RHS
->
getLocEnd
()));
string
text
=
"bpf_dins_pkt(skb, (u64)"
+
base
+
"+"
+
to_string
(
ofs
>>
3
)
+
", "
+
to_string
(
ofs
&
0x7
)
+
", "
+
to_string
(
sz
)
+
", "
+
rhs
+
")"
;
rewriter_
.
ReplaceText
(
SourceRange
(
E
->
getLocStart
(),
E
->
getLocEnd
()),
text
);
}
}
}
}
}
return
true
;
}
bool
BTypeVisitor
::
VisitImplicitCastExpr
(
ImplicitCastExpr
*
E
)
{
// use dext only for RValues
if
(
E
->
getCastKind
()
!=
CK_LValueToRValue
)
return
true
;
MemberExpr
*
Memb
=
dyn_cast
<
MemberExpr
>
(
E
->
IgnoreImplicit
());
if
(
!
Memb
)
return
true
;
Expr
*
Base
=
Memb
->
getBase
()
->
IgnoreImplicit
();
if
(
DeclRefExpr
*
Ref
=
dyn_cast
<
DeclRefExpr
>
(
Base
))
{
if
(
DeclRefExpr
*
Ref
=
dyn_cast
<
DeclRefExpr
>
(
Base
))
{
if
(
DeprecatedAttr
*
A
=
Ref
->
getDecl
()
->
getAttr
<
DeprecatedAttr
>
())
{
if
(
DeprecatedAttr
*
A
=
Ref
->
getDecl
()
->
getAttr
<
DeprecatedAttr
>
())
{
if
(
A
->
getMessage
()
==
"packet"
)
{
if
(
A
->
getMessage
()
==
"packet"
)
{
if
(
FieldDecl
*
F
=
dyn_cast
<
FieldDecl
>
(
E
->
getMemberDecl
()))
{
if
(
FieldDecl
*
F
=
dyn_cast
<
FieldDecl
>
(
Memb
->
getMemberDecl
()))
{
uint64_t
ofs
=
C
.
getFieldOffset
(
F
);
uint64_t
ofs
=
C
.
getFieldOffset
(
F
);
uint64_t
sz
=
F
->
isBitField
()
?
F
->
getBitWidthValue
(
C
)
:
C
.
getTypeSize
(
F
->
getType
());
uint64_t
sz
=
F
->
isBitField
()
?
F
->
getBitWidthValue
(
C
)
:
C
.
getTypeSize
(
F
->
getType
());
string
base
=
rewriter_
.
getRewrittenText
(
SourceRange
(
Base
->
getLocStart
(),
Base
->
getLocEnd
()));
string
base
=
rewriter_
.
getRewrittenText
(
SourceRange
(
Base
->
getLocStart
(),
Base
->
getLocEnd
()));
...
...
src/cc/b_frontend_action.h
View file @
f197faf4
...
@@ -40,7 +40,8 @@ class BTypeVisitor : public clang::RecursiveASTVisitor<BTypeVisitor> {
...
@@ -40,7 +40,8 @@ class BTypeVisitor : public clang::RecursiveASTVisitor<BTypeVisitor> {
bool
VisitCallExpr
(
clang
::
CallExpr
*
Call
);
bool
VisitCallExpr
(
clang
::
CallExpr
*
Call
);
bool
VisitVarDecl
(
clang
::
VarDecl
*
Decl
);
bool
VisitVarDecl
(
clang
::
VarDecl
*
Decl
);
bool
VisitArraySubscriptExpr
(
clang
::
ArraySubscriptExpr
*
E
);
bool
VisitArraySubscriptExpr
(
clang
::
ArraySubscriptExpr
*
E
);
bool
VisitMemberExpr
(
clang
::
MemberExpr
*
E
);
bool
VisitBinaryOperator
(
clang
::
BinaryOperator
*
E
);
bool
VisitImplicitCastExpr
(
clang
::
ImplicitCastExpr
*
E
);
private:
private:
clang
::
ASTContext
&
C
;
clang
::
ASTContext
&
C
;
...
...
src/cc/bpf_helpers.h
View file @
f197faf4
...
@@ -254,6 +254,11 @@ int bpf_l4_csum_replace_(void *ctx, u64 off, u64 from, u64 to, u64 flags) {
...
@@ -254,6 +254,11 @@ int bpf_l4_csum_replace_(void *ctx, u64 off, u64 from, u64 to, u64 flags) {
return
bpf_l4_csum_replace
(
ctx
,
off
,
from
,
to
,
flags
);
return
bpf_l4_csum_replace
(
ctx
,
off
,
from
,
to
,
flags
);
}
}
#define incr_cksum_l3(expr, oldval, newval) \
bpf_l3_csum_replace_(skb, (u64)expr, oldval, newval, sizeof(newval))
#define incr_cksum_l4(expr, oldval, newval, is_pseudo) \
bpf_l4_csum_replace_(skb, (u64)expr, oldval, newval, ((is_pseudo & 0x1) << 4) | sizeof(newval))
#define lock_xadd(ptr, val) ((void)__sync_fetch_and_add(ptr, val))
#define lock_xadd(ptr, val) ((void)__sync_fetch_and_add(ptr, val))
#endif
#endif
src/cc/proto.h
View file @
f197faf4
...
@@ -7,41 +7,41 @@ struct ethernet_t {
...
@@ -7,41 +7,41 @@ struct ethernet_t {
}
__attribute__
((
packed
));
}
__attribute__
((
packed
));
struct
dot1q_t
{
struct
dot1q_t
{
u
32
pri
:
3
;
u
16
pri
:
3
;
u
32
cfi
:
1
;
u
16
cfi
:
1
;
u
32
vlanid
:
12
;
u
16
vlanid
:
12
;
u
32
type
:
16
;
u
16
type
;
}
__attribute__
((
packed
));
}
__attribute__
((
packed
));
struct
ip_t
{
struct
ip_t
{
u
32
ver
:
4
;
// byte 0
u
8
ver
:
4
;
// byte 0
u
32
hlen
:
4
;
u
8
hlen
:
4
;
u
32
tos
:
8
;
u
8
tos
;
u
32
tlen
:
16
;
u
16
tlen
;
u
32
identification
:
16
;
// byte 4
u
16
identification
;
// byte 4
u
32
ffo_unused
:
1
;
u
16
ffo_unused
:
1
;
u
32
df
:
1
;
u
16
df
:
1
;
u
32
mf
:
1
;
u
16
mf
:
1
;
u
32
foffset
:
13
;
u
16
foffset
:
13
;
u
32
ttl
:
8
;
// byte 8
u
8
ttl
;
// byte 8
u
32
nextp
:
8
;
u
8
nextp
;
u
32
hchecksum
:
16
;
u
16
hchecksum
;
u32
src
:
32
;
// byte 12
u32
src
;
// byte 12
u32
dst
:
32
;
// byte 16
u32
dst
;
// byte 16
}
__attribute__
((
packed
));
}
__attribute__
((
packed
));
struct
udp_t
{
struct
udp_t
{
u
32
sport
:
16
;
u
16
sport
;
u
32
dport
:
16
;
u
16
dport
;
u
32
length
:
16
;
u
16
length
;
u
32
crc
:
16
;
u
16
crc
;
}
__attribute__
((
packed
));
}
__attribute__
((
packed
));
struct
tcp_t
{
struct
tcp_t
{
u16
src_port
:
16
;
// byte 0
u16
src_port
;
// byte 0
u16
dst_port
:
16
;
u16
dst_port
;
u32
seq_num
:
32
;
// byte 4
u32
seq_num
;
// byte 4
u32
ack_num
:
32
;
// byte 8
u32
ack_num
;
// byte 8
u8
offset
:
4
;
// byte 12
u8
offset
:
4
;
// byte 12
u8
reserved
:
4
;
u8
reserved
:
4
;
u8
flag_cwr
:
1
;
u8
flag_cwr
:
1
;
...
@@ -52,7 +52,7 @@ struct tcp_t {
...
@@ -52,7 +52,7 @@ struct tcp_t {
u8
flag_rst
:
1
;
u8
flag_rst
:
1
;
u8
flag_syn
:
1
;
u8
flag_syn
:
1
;
u8
flag_fin
:
1
;
u8
flag_fin
:
1
;
u16
rcv_wnd
:
16
;
u16
rcv_wnd
;
u16
cksum
:
16
;
// byte 16
u16
cksum
;
// byte 16
u16
urg_ptr
:
16
;
u16
urg_ptr
;
}
__attribute__
((
packed
));
}
__attribute__
((
packed
));
tests/jit/CMakeLists.txt
View file @
f197faf4
...
@@ -2,8 +2,10 @@ add_test(NAME py_test1_b WORKING_DIRECTORY ${CMAKE_CURRENT_SOURCE_DIR}
...
@@ -2,8 +2,10 @@ add_test(NAME py_test1_b WORKING_DIRECTORY ${CMAKE_CURRENT_SOURCE_DIR}
COMMAND
${
TEST_WRAPPER
}
py_test1_b namespace
${
CMAKE_CURRENT_SOURCE_DIR
}
/test1.py test1.b proto.b
)
COMMAND
${
TEST_WRAPPER
}
py_test1_b namespace
${
CMAKE_CURRENT_SOURCE_DIR
}
/test1.py test1.b proto.b
)
add_test
(
NAME py_test1_c WORKING_DIRECTORY
${
CMAKE_CURRENT_SOURCE_DIR
}
add_test
(
NAME py_test1_c WORKING_DIRECTORY
${
CMAKE_CURRENT_SOURCE_DIR
}
COMMAND
${
TEST_WRAPPER
}
py_test1_c namespace
${
CMAKE_CURRENT_SOURCE_DIR
}
/test1.py test1.c
)
COMMAND
${
TEST_WRAPPER
}
py_test1_c namespace
${
CMAKE_CURRENT_SOURCE_DIR
}
/test1.py test1.c
)
add_test
(
NAME py_test2 WORKING_DIRECTORY
${
CMAKE_CURRENT_SOURCE_DIR
}
add_test
(
NAME py_test2_b WORKING_DIRECTORY
${
CMAKE_CURRENT_SOURCE_DIR
}
COMMAND
${
TEST_WRAPPER
}
py_test2 namespace
${
CMAKE_CURRENT_SOURCE_DIR
}
/test2.py test2.b proto.b
)
COMMAND
${
TEST_WRAPPER
}
py_test2_b namespace
${
CMAKE_CURRENT_SOURCE_DIR
}
/test2.py test2.b proto.b
)
add_test
(
NAME py_test2_c WORKING_DIRECTORY
${
CMAKE_CURRENT_SOURCE_DIR
}
COMMAND
${
TEST_WRAPPER
}
py_test2_c namespace
${
CMAKE_CURRENT_SOURCE_DIR
}
/test2.py test2.c
)
add_test
(
NAME py_test3 WORKING_DIRECTORY
${
CMAKE_CURRENT_SOURCE_DIR
}
add_test
(
NAME py_test3 WORKING_DIRECTORY
${
CMAKE_CURRENT_SOURCE_DIR
}
COMMAND
${
TEST_WRAPPER
}
py_test3 namespace
${
CMAKE_CURRENT_SOURCE_DIR
}
/test3.py test3.c
)
COMMAND
${
TEST_WRAPPER
}
py_test3 namespace
${
CMAKE_CURRENT_SOURCE_DIR
}
/test3.py test3.c
)
add_test
(
NAME py_trace1 WORKING_DIRECTORY
${
CMAKE_CURRENT_SOURCE_DIR
}
add_test
(
NAME py_trace1 WORKING_DIRECTORY
${
CMAKE_CURRENT_SOURCE_DIR
}
...
...
tests/jit/test1.b
View file @
f197faf4
...
@@ -12,7 +12,7 @@ struct skbuff {
...
@@ -12,7 +12,7 @@ struct skbuff {
u32 type:32;
u32 type:32;
};
};
u32
main
(struct skbuff *skb) {
u32
on_packet
(struct skbuff *skb) {
u32 ret:32 = 0;
u32 ret:32 = 0;
goto proto::ethernet;
goto proto::ethernet;
...
...
tests/jit/test1.c
View file @
f197faf4
...
@@ -12,8 +12,8 @@ struct IPLeaf {
...
@@ -12,8 +12,8 @@ struct IPLeaf {
BPF_TABLE
(
"hash"
,
struct
IPKey
,
struct
IPLeaf
,
stats
,
256
);
BPF_TABLE
(
"hash"
,
struct
IPKey
,
struct
IPLeaf
,
stats
,
256
);
BPF_EXPORT
(
main
)
BPF_EXPORT
(
on_packet
)
int
_main
(
struct
__sk_buff
*
skb
)
{
int
on_packet
(
struct
__sk_buff
*
skb
)
{
BEGIN
(
ethernet
);
BEGIN
(
ethernet
);
PROTO
(
ethernet
)
{
PROTO
(
ethernet
)
{
...
...
tests/jit/test1.py
View file @
f197faf4
...
@@ -25,8 +25,8 @@ class Leaf(Structure):
...
@@ -25,8 +25,8 @@ class Leaf(Structure):
class
TestBPFSocket
(
TestCase
):
class
TestBPFSocket
(
TestCase
):
def
setUp
(
self
):
def
setUp
(
self
):
b
=
BPF
(
arg1
,
arg2
,
debug
=
0
)
b
=
BPF
(
arg1
,
arg2
,
debug
=
0
)
fn
=
b
.
load_func
(
"
main
"
,
BPF
.
SOCKET_FILTER
)
fn
=
b
.
load_func
(
"
on_packet
"
,
BPF
.
SOCKET_FILTER
)
BPF
.
attach_socket
(
fn
,
"eth0"
)
BPF
.
attach_
raw_
socket
(
fn
,
"eth0"
)
self
.
stats
=
b
.
get_table
(
"stats"
,
Key
,
Leaf
)
self
.
stats
=
b
.
get_table
(
"stats"
,
Key
,
Leaf
)
def
test_ping
(
self
):
def
test_ping
(
self
):
...
...
tests/jit/test2.b
View file @
f197faf4
...
@@ -17,7 +17,7 @@ struct skbuff {
...
@@ -17,7 +17,7 @@ struct skbuff {
u32 type:32;
u32 type:32;
};
};
u32
main
(struct skbuff *skb) {
u32
on_packet
(struct skbuff *skb) {
u32 ret:32 = 1;
u32 ret:32 = 1;
u32 orig_dip:32 = 0;
u32 orig_dip:32 = 0;
...
...
tests/jit/test2.c
0 → 100644
View file @
f197faf4
#include "../../src/cc/bpf_helpers.h"
#include "../../src/cc/proto.h"
struct
IPKey
{
u32
dip
;
u32
sip
;
};
struct
IPLeaf
{
u32
xdip
;
u32
xsip
;
u64
xlated_pkts
;
};
BPF_TABLE
(
"hash"
,
struct
IPKey
,
struct
IPLeaf
,
xlate
,
1024
);
BPF_EXPORT
(
on_packet
)
int
on_packet
(
struct
__sk_buff
*
skb
)
{
BEGIN
(
ethernet
);
u32
orig_dip
=
0
;
u32
orig_sip
=
0
;
struct
IPLeaf
*
xleaf
;
PROTO
(
ethernet
)
{
switch
(
ethernet
->
type
)
{
case
0x0800
:
goto
ip
;
case
0x8100
:
goto
dot1q
;
}
goto
EOP
;
}
PROTO
(
dot1q
)
{
switch
(
dot1q
->
type
)
{
case
0x0800
:
goto
ip
;
}
goto
EOP
;
}
PROTO
(
ip
)
{
orig_dip
=
ip
->
dst
;
orig_sip
=
ip
->
src
;
struct
IPKey
key
=
{.
dip
=
orig_dip
,
.
sip
=
orig_sip
};
xleaf
=
xlate
.
lookup
(
&
key
);
if
(
xleaf
)
{
ip
->
dst
=
xleaf
->
xdip
;
incr_cksum_l3
(
&
ip
->
hchecksum
,
orig_dip
,
xleaf
->
xdip
);
ip
->
src
=
xleaf
->
xsip
;
incr_cksum_l3
(
&
ip
->
hchecksum
,
orig_sip
,
xleaf
->
xsip
);
lock_xadd
(
&
xleaf
->
xlated_pkts
,
1
);
}
switch
(
ip
->
nextp
)
{
case
6
:
goto
tcp
;
case
17
:
goto
udp
;
}
goto
EOP
;
}
PROTO
(
udp
)
{
if
(
xleaf
)
{
incr_cksum_l4
(
&
udp
->
crc
,
orig_dip
,
xleaf
->
xdip
,
1
);
incr_cksum_l4
(
&
udp
->
crc
,
orig_sip
,
xleaf
->
xsip
,
1
);
}
goto
EOP
;
}
PROTO
(
tcp
)
{
if
(
xleaf
)
{
incr_cksum_l4
(
&
tcp
->
cksum
,
orig_dip
,
xleaf
->
xdip
,
1
);
incr_cksum_l4
(
&
tcp
->
cksum
,
orig_sip
,
xleaf
->
xsip
,
1
);
}
goto
EOP
;
}
EOP:
return
0
;
}
tests/jit/test2.py
View file @
f197faf4
...
@@ -23,8 +23,8 @@ class Leaf(Structure):
...
@@ -23,8 +23,8 @@ class Leaf(Structure):
class
TestBPFSocket
(
TestCase
):
class
TestBPFSocket
(
TestCase
):
def
setUp
(
self
):
def
setUp
(
self
):
b
=
BPF
(
arg1
,
arg2
,
debug
=
0
)
b
=
BPF
(
arg1
,
arg2
,
debug
=
1
)
fn
=
b
.
load_func
(
"
main
"
,
BPF
.
SCHED_CLS
)
fn
=
b
.
load_func
(
"
on_packet
"
,
BPF
.
SCHED_CLS
)
BPF
.
attach_classifier
(
fn
,
"eth0"
)
BPF
.
attach_classifier
(
fn
,
"eth0"
)
self
.
xlate
=
b
.
get_table
(
"xlate"
,
Key
,
Leaf
)
self
.
xlate
=
b
.
get_table
(
"xlate"
,
Key
,
Leaf
)
...
...
tests/wrapper.sh.in
View file @
f197faf4
...
@@ -28,6 +28,7 @@ function ns_run() {
...
@@ -28,6 +28,7 @@ function ns_run() {
sudo
ip netns
exec
$ns
tc qdisc add dev eth0 root prio
sudo
ip netns
exec
$ns
tc qdisc add dev eth0 root prio
sudo
ip netns
exec
$ns
ip addr add dev eth0 172.16.1.2/24
sudo
ip netns
exec
$ns
ip addr add dev eth0 172.16.1.2/24
sudo
ip netns
exec
$ns
ip
link set
eth0 up
sudo
ip netns
exec
$ns
ip
link set
eth0 up
sudo
ip netns
exec
$ns
ethtool
-K
eth0 tx off
sudo
ip addr add dev
$ns
.out 172.16.1.1/24
sudo
ip addr add dev
$ns
.out 172.16.1.1/24
sudo
ip
link set
$ns
.out up
sudo
ip
link set
$ns
.out up
sudo
bash
-c
"PYTHONPATH=
$PYTHONPATH
LD_LIBRARY_PATH=
$LD_LIBRARY_PATH
ip netns exec
$ns
$cmd
$1
$2
"
sudo
bash
-c
"PYTHONPATH=
$PYTHONPATH
LD_LIBRARY_PATH=
$LD_LIBRARY_PATH
ip netns exec
$ns
$cmd
$1
$2
"
...
...
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