mirror of
https://github.com/samba-team/samba.git
synced 2025-12-17 04:23:50 +03:00
- properly support --samba3-header argument used alone - support `security_descriptor' data type - only print pidl warnings, not perl warnings on erratic input - insert copyright header in templates
224 lines
4.7 KiB
Perl
224 lines
4.7 KiB
Perl
###################################################
|
|
# Samba3 NDR header generator for IDL structures
|
|
# Copyright jelmer@samba.org 2005
|
|
# released under the GNU GPL
|
|
|
|
package Parse::Pidl::Samba3::Header;
|
|
|
|
use strict;
|
|
use Parse::Pidl::Typelist qw(hasType getType);
|
|
use Parse::Pidl::Util qw(has_property ParseExpr);
|
|
use Parse::Pidl::NDR qw(GetPrevLevel GetNextLevel ContainsDeferred);
|
|
use Parse::Pidl::Samba3::Types qw(DeclShort StringType);
|
|
|
|
use vars qw($VERSION);
|
|
$VERSION = '0.01';
|
|
|
|
my $res = "";
|
|
my $tabs = "";
|
|
sub indent() { $tabs.="\t"; }
|
|
sub deindent() { $tabs = substr($tabs, 1); }
|
|
sub pidl($) { $res .= $tabs.(shift)."\n"; }
|
|
sub fatal($$) { my ($e,$s) = @_; die("$e->{ORIGINAL}->{FILE}:$e->{ORIGINAL}->{LINE}: $s\n"); }
|
|
sub warning($$) { my ($e,$s) = @_; warn("$e->{ORIGINAL}->{FILE}:$e->{ORIGINAL}->{LINE}: $s\n"); }
|
|
|
|
sub ParseElement($)
|
|
{
|
|
my $e = shift;
|
|
|
|
foreach my $l (@{$e->{LEVELS}}) {
|
|
if ($l->{TYPE} eq "POINTER") {
|
|
next if ($l->{POINTER_TYPE} eq "ref" and $l->{LEVEL} eq "TOP");
|
|
pidl "\tuint32 ptr$l->{POINTER_INDEX}_$e->{NAME};";
|
|
} elsif ($l->{TYPE} eq "SWITCH") {
|
|
} elsif ($l->{TYPE} eq "DATA") {
|
|
my $n = DeclShort($e);
|
|
pidl "\t$n;" if ($n);
|
|
} elsif ($l->{TYPE} eq "ARRAY" and $l->{IS_ZERO_TERMINATED}) {
|
|
my ($t,$f) = StringType($e,$l);
|
|
pidl "\t" . uc($t) . " $e->{NAME};";
|
|
return;
|
|
} elsif ($l->{TYPE} eq "ARRAY") {
|
|
if ($l->{IS_CONFORMANT}) {
|
|
pidl "\tuint32 size_$e->{NAME};";
|
|
}
|
|
if ($l->{IS_VARYING}) {
|
|
pidl "\tuint32 length_$e->{NAME};";
|
|
pidl "\tuint32 offset_$e->{NAME};";
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
sub CreateStruct($$$$)
|
|
{
|
|
my ($if,$fn,$n,$t) = @_;
|
|
|
|
pidl "typedef struct $n {";
|
|
ParseElement($_) foreach (@$t);
|
|
|
|
if (not @$t) {
|
|
# Some compilers don't like empty structs
|
|
pidl "\tuint32 dummy;";
|
|
}
|
|
|
|
pidl "} " . uc($n) . ";";
|
|
pidl "";
|
|
}
|
|
|
|
sub ParseFunction($$)
|
|
{
|
|
my ($if,$fn) = @_;
|
|
|
|
my @in = ();
|
|
my @out = ();
|
|
|
|
foreach (@{$fn->{ELEMENTS}}) {
|
|
push (@in, $_) if (grep(/in/, @{$_->{DIRECTION}}));
|
|
push (@out, $_) if (grep(/out/, @{$_->{DIRECTION}}));
|
|
}
|
|
|
|
if (defined($fn->{RETURN_TYPE})) {
|
|
push (@out, {
|
|
NAME => "status",
|
|
TYPE => $fn->{RETURN_TYPE},
|
|
LEVELS => [
|
|
{
|
|
TYPE => "DATA",
|
|
DATA_TYPE => $fn->{RETURN_TYPE}
|
|
}
|
|
]
|
|
} );
|
|
}
|
|
|
|
# define Q + R structures for functions
|
|
|
|
CreateStruct($if, $fn, "$if->{NAME}_q_$fn->{NAME}", \@in);
|
|
CreateStruct($if, $fn, "$if->{NAME}_r_$fn->{NAME}", \@out);
|
|
}
|
|
|
|
sub ParseStruct($$$)
|
|
{
|
|
my ($if,$s,$n) = @_;
|
|
|
|
CreateStruct($if, $s, "$if->{NAME}_$n", $s->{ELEMENTS});
|
|
}
|
|
|
|
sub ParseUnion($$$)
|
|
{
|
|
my ($if,$u,$n) = @_;
|
|
|
|
my $extra = {
|
|
switch_value => $u->{SWITCH_TYPE}
|
|
};
|
|
|
|
if (not defined($extra->{switch_value})) {
|
|
$extra->{switch_value} = "uint32";
|
|
}
|
|
|
|
foreach my $e (@{$u->{ELEMENTS}}) {
|
|
foreach my $l (@{$e->{LEVELS}}) {
|
|
if ($l->{TYPE} eq "ARRAY") {
|
|
if ($l->{IS_CONFORMANT}) {
|
|
$extra->{"size"} = "uint32";
|
|
}
|
|
if ($l->{IS_VARYING}) {
|
|
$extra->{"length"} = $extra->{"offset"} = "uint32";
|
|
}
|
|
} elsif ($l->{TYPE} eq "POINTER") {
|
|
$extra->{"ptr$l->{POINTER_INDEX}"} = "uint32";
|
|
} elsif ($l->{TYPE} eq "SWITCH") {
|
|
$extra->{"level"} = "uint32";
|
|
}
|
|
}
|
|
}
|
|
|
|
pidl "typedef struct $if->{NAME}_$n\_ctr {";
|
|
indent;
|
|
pidl "$extra->{$_} $_;" foreach (keys %$extra);
|
|
pidl "union $if->{NAME}_$n {";
|
|
indent;
|
|
foreach (@{$u->{ELEMENTS}}) {
|
|
next if ($_->{TYPE} eq "EMPTY");
|
|
pidl "\t" . DeclShort($_) . ";";
|
|
}
|
|
deindent;
|
|
pidl "} u;";
|
|
deindent;
|
|
pidl "} ".uc("$if->{NAME}_$n\_ctr") .";";
|
|
pidl "";
|
|
}
|
|
|
|
sub ParseEnum($$$)
|
|
{
|
|
my ($if,$s,$n) = @_;
|
|
|
|
pidl "typedef enum {";
|
|
pidl "$_," foreach (@{$s->{ELEMENTS}});
|
|
pidl "} $n;";
|
|
}
|
|
|
|
sub ParseBitmap($$$)
|
|
{
|
|
my ($if,$s,$n) = @_;
|
|
|
|
pidl "#define $_" foreach (@{$s->{ELEMENTS}});
|
|
}
|
|
|
|
sub ParseInterface($)
|
|
{
|
|
my $if = shift;
|
|
|
|
my $def = "_RPC_" . uc($if->{NAME}) . "_H";
|
|
|
|
pidl "";
|
|
|
|
pidl "\#ifndef $def";
|
|
pidl "\#define $def";
|
|
|
|
pidl "";
|
|
|
|
foreach (@{$if->{FUNCTIONS}}) {
|
|
pidl "\#define " . uc($_->{NAME}) . " $_->{OPNUM}" ;
|
|
}
|
|
|
|
pidl "";
|
|
|
|
foreach (@{$if->{TYPES}}) {
|
|
ParseStruct($if, $_->{DATA}, $_->{NAME}) if ($_->{DATA}->{TYPE} eq "STRUCT");
|
|
ParseEnum($if, $_->{DATA}, $_->{NAME}) if ($_->{DATA}->{TYPE} eq "ENUM");
|
|
ParseBitmap($if, $_->{DATA}, $_->{NAME}) if ($_->{DATA}->{TYPE} eq "BITMAP");
|
|
ParseUnion($if, $_->{DATA}, $_->{NAME}) if ($_->{DATA}->{TYPE} eq "UNION");
|
|
}
|
|
|
|
ParseFunction($if, $_) foreach (@{$if->{FUNCTIONS}});
|
|
|
|
foreach (@{$if->{CONSTS}}) {
|
|
pidl "$_->{NAME} ($_->{VALUE})";
|
|
}
|
|
|
|
pidl "\#endif /* $def */";
|
|
}
|
|
|
|
sub Parse($$)
|
|
{
|
|
my($ndr,$filename) = @_;
|
|
|
|
$res = "";
|
|
$tabs = "";
|
|
|
|
pidl "/*";
|
|
pidl " * Unix SMB/CIFS implementation.";
|
|
pidl " * header auto-generated by pidl. DO NOT MODIFY!";
|
|
pidl " */";
|
|
pidl "";
|
|
|
|
# Loop over interfaces
|
|
foreach (@{$ndr}) {
|
|
ParseInterface($_) if ($_->{TYPE} eq "INTERFACE");
|
|
}
|
|
return $res;
|
|
}
|
|
|
|
1;
|