File Coverage

blib/lib/Protocol/DBus/Parser.pm
Criterion Covered Total %
statement 57 78 73.0
branch 17 46 36.9
condition 2 14 14.2
subroutine 10 11 90.9
pod 0 3 0.0
total 86 152 56.5


line stmt bran cond sub pod time code
1             package Protocol::DBus::Parser;
2              
3 4     4   23 use strict;
  4         5  
  4         92  
4 4     4   17 use warnings;
  4         4  
  4         67  
5              
6 4     4   16 use Protocol::DBus::Marshal ();
  4         4  
  4         37  
7 4     4   12 use Protocol::DBus::Message ();
  4         5  
  4         72  
8 4     4   1207 use Protocol::DBus::X ();
  4         11  
  4         117  
9              
10 4     4   24 use constant SINGLE_UNIX_FD_CMSGHDR => (0, 0, pack 'I!');
  4         5  
  4         204  
11              
12 4     4   20 use constant _LE_INIT_UNPACK => 'x4 V x4 V';
  4         4  
  4         137  
13 4     4   19 use constant _BE_INIT_UNPACK => 'x4 N x4 N';
  4         5  
  4         2009  
14              
15             sub new {
16 3     3 0 20 my ($class, $socket) = @_;
17              
18 3         58 return bless { _s => $socket, _buf => q<> }, $class;
19             }
20              
21             sub get_message {
22 3     3 0 8 my ($self) = @_;
23              
24 3         5 my $msg;
25              
26 3 50       47 if (!$self->{'_bodysz'}) {
27 3 50 0     20535 if (defined recv( $self->{'_s'}, my $peek, 16, Socket::MSG_PEEK() )) {
    0          
28 3 50       33 if (!length $peek) {
    50          
29 0         0 die Protocol::DBus::X->create('SurpriseShutdown');
30             }
31             elsif ( 16 == length $peek ) {
32 3 50       26 @{$self}{'_bodysz', '_hdrsz'} = unpack(
  3         114  
33             (0 == index($peek, 'B')) ? _BE_INIT_UNPACK() : _LE_INIT_UNPACK(),
34             $peek,
35             );
36              
37 3         37 Protocol::DBus::Pack::align( $self->{'_hdrsz'}, 8 );
38              
39 3         15 $self->{'_msgsz'} = 16 + $self->{'_hdrsz'} + $self->{'_bodysz'};
40             }
41             }
42             elsif (!$!{'EAGAIN'} && !$!{'EWOULDBLOCK'}) {
43 0         0 die "recv(): $!";
44             }
45             }
46              
47 3 50 33     55 if (defined $self->{'_bodysz'} && !defined $self->{'_unix_fds'}) {
48 3 50 0     40 if (defined recv( $self->{'_s'}, my $full_hdr, 16 + $self->{'_hdrsz'}, Socket::MSG_PEEK() )) {
    0          
49 3 50       23 if ( length($full_hdr) == 16 + $self->{'_hdrsz'} ) {
50 3         28 my ($hdr) = Protocol::DBus::Message::Header::parse_simple(\$full_hdr);
51              
52 3   50     42 $self->{'_unix_fds'} = $hdr->[6]{ Protocol::DBus::Message::Header::FIELD()->{'UNIX_FDS'} } || 0;
53              
54 3         8 $self->{'_pending_unix_fds'} = $self->{'_unix_fds'};
55              
56 3         5 my $body_sig = $hdr->[6]{ Protocol::DBus::Message::Header::FIELD()->{'SIGNATURE'} };
57              
58 3 100       20 if ($hdr->[4]) {
59 2 50       9 die "No SIGNATURE header field!" if !defined $body_sig;
60             }
61             }
62             }
63             elsif (!$!{'EAGAIN'} && !$!{'EWOULDBLOCK'}) {
64 0         0 die "recv(): $!";
65             }
66             }
67              
68 3 50       255 if (defined $self->{'_unix_fds'}) {
69              
70 3         9 my $needed_bytes = $self->{'_msgsz'} - length $self->{'_buf'};
71              
72 3         4 my $got;
73              
74 3 50       9 if ($self->{'_unix_fds'}) {
75 0         0 my $msg = Socket::MsgHdr->new(
76             buflen => $needed_bytes,
77             );
78              
79             # The unix FDs might arrive in a single control
80             # message, as individual control messages, or as
81             # some combination thereof. There is no way to know.
82             # So plan for the worst, and assume each unix FD is
83             # in its own control.
84 0         0 $msg->cmsghdr( (SINGLE_UNIX_FD_CMSGHDR()) x $self->{'_pending_unix_fds'} );
85              
86 0         0 $got = Socket::MsgHdr::recvmsg( $self->{'_s'}, $msg );
87 0 0       0 if (defined $got) {
88              
89 0 0       0 if ($self->{'_pending_unix_fds'}) {
90 0         0 require Protocol::DBus::Parser::UnixFDs;
91 0         0 push @{ $self->{'_filehandles'} }, Protocol::DBus::Parser::UnixFDs::extract_from_msghdr($msg);
  0         0  
92 0         0 $self->{'_pending_unix_fds'} = $self->{'_unix_fds'} - @{ $self->{'_filehandles'} };
  0         0  
93             }
94              
95 0         0 $self->{'_buf'} .= $msg->buf();
96             }
97             }
98             else {
99             $got = sysread(
100             $self->{'_s'},
101             $self->{'_buf'},
102             $needed_bytes,
103 3         59 length $self->{'_buf'},
104             );
105             }
106              
107 3 50 0     10 if (defined $got) {
    0          
108 3 50       8 if ($got >= $needed_bytes) {
    0          
109 3 50       10 local $Protocol::DBus::Marshal::PRESERVE_VARIANT_SIGNATURES = 1 if $self->{'_preserve_variant_signatures'};
110              
111             # This clears out the buffer .. it should??
112 3         26 my $msg = Protocol::DBus::Message->parse( \$self->{'_buf'}, delete $self->{'_filehandles'} );
113              
114 3 50       13 die "Not enough bytes??" if !$msg;
115              
116 3         5 delete @{$self}{'_bodysz', '_unix_fds'};
  3         7  
117              
118 3         10 return $msg;
119             }
120             elsif (!$got) {
121 0           die "Peer stopped writing!";
122             }
123             }
124             elsif (!$!{'EAGAIN'} && !$!{'EWOULDBLOCK'}) {
125 0           die "recv(): $!";
126             }
127             }
128              
129 0           return undef;
130             }
131              
132             sub preserve_variant_signatures {
133 0     0 0   my $self = shift;
134              
135 0 0         if (@_) {
136 0           $self->{'_preserve_variant_signatures'} = !!$_[0];
137             }
138              
139 0           return !!$self->{'_preserve_variant_signatures'};
140             }
141              
142             #sub DESTROY {
143             # print "DESTROYED: [$_[0]]\n";
144             #}
145              
146             1;