1
0
mirror of https://github.com/samba-team/samba.git synced 2025-01-12 09:18:10 +03:00

Merge branch 'master' of ssh://jra@git.samba.org/data/git/samba

This commit is contained in:
Jeremy Allison 2009-04-09 22:46:59 -07:00
commit 7f10bf980e
230 changed files with 14184 additions and 22627 deletions

View File

@ -8,7 +8,7 @@
<manvolnum>8</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">System Administration tools</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>
<refnamediv>

View File

@ -7,7 +7,7 @@
<manvolnum>8</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">System Administration tools</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>1</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">User Commands</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>8</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">System Administration tools</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>8</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">System Administration tools</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>8</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">System Administration tools</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>8</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">System Administration tools</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>8</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">System Administration tools</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>8</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">System Administration tools</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>8</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">System Administration tools</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>8</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">System Administration tools</refmiscinfo>
<refmiscinfo class="version">3.3</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>3</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">C Library Functions</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>
<refnamediv>

View File

@ -7,7 +7,7 @@
<manvolnum>1</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">User Commands</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>1</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">User Commands</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>1</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">User Commands</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>1</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">User Commands</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>1</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">User Commands</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>7</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">7</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>5</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">File Formats and Conventions</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>1</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">User Commands</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>8</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">System Administration tools</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>8</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">System Administration tools</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>8</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">System Administration tools</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>1</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">User Commands</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>1</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">User Commands</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>7</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">7</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>8</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">System Administration tools</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>1</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">User Commands</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>1</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">User Commands</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>7</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">Miscellanea</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>1</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">User Commands</refmiscinfo>
<refmiscinfo class="version">3.3</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -6,7 +6,7 @@
<manvolnum>5</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">File Formats and Conventions</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>1</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">User Commands</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>1</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">User Commands</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>1</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">User Commands</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>
@ -66,7 +66,7 @@
<para>The <parameter>all</parameter> destination causes the
message to "broadcast" to all running daemons including nmbd and
winbind. This is a change for Samba 3.3, prior to this the
paramter smbd used to do this.</para>
parameter smbd used to do this.</para>
<para>The <parameter>smbd</parameter> destination causes the
message to be sent to the smbd daemon specified in the

View File

@ -7,7 +7,7 @@
<manvolnum>1</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">User Commands</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>8</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">System Administration tools</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>1</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">User Commands</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>5</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">File Formats and Conventions</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>5</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">File Formats and Conventions</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>8</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">System Administration tools</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>8</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">System Administration tools</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>1</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">User Commands</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>1</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">User Commands</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>1</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">User Commands</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>8</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">System Administration tools</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>8</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">System Administration tools</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>8</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">System Administration tools</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>8</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">System Administration tools</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>1</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">User Commands</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>8</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">System Administration tools</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>8</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">System Administration tools</refmiscinfo>
<refmiscinfo class="version">3.3</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>8</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">System Administration tools</refmiscinfo>
<refmiscinfo class="version">3.3</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>8</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">System Administration tools</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>8</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">System Administration tools</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>8</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">System Administration tools</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>8</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">System Administration tools</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>8</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">System Administration tools</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>8</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">System Administration tools</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>8</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">System Administration tools</refmiscinfo>
<refmiscinfo class="version">3.3</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>8</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">System Administration tools</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>8</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">System Administration tools</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>8</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">System Administration tools</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>8</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">System Administration tools</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>8</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">System Administration tools</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>8</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">System Administration tools</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>8</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">System Administration tools</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>8</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">System Administration tools</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>8</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">System Administration tools</refmiscinfo>
<refmiscinfo class="version">3.3</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>
<refnamediv>

View File

@ -7,7 +7,7 @@
<manvolnum>8</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">System Administration tools</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>8</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">System Administration tools</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>8</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">System Administration tools</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>8</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">System Administration tools</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>8</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">System Administration tools</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>8</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">System Administration tools</refmiscinfo>
<refmiscinfo class="version">3.3</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>8</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">System Administration tools</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>8</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">System Administration tools</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>8</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">System Administration tools</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>1</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">User Commands</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>1</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">User Commands</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>7</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">7</refmiscinfo>
<refmiscinfo class="version">3.3</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -7,7 +7,7 @@
<manvolnum>8</manvolnum>
<refmiscinfo class="source">Samba</refmiscinfo>
<refmiscinfo class="manual">System Administration tools</refmiscinfo>
<refmiscinfo class="version">3.2</refmiscinfo>
<refmiscinfo class="version">3.5</refmiscinfo>
</refmeta>

View File

@ -204,7 +204,7 @@
; comment = Public Stuff
; path = /home/samba
; public = yes
; writable = yes
; writable = no
; printable = no
; write list = @staff

View File

@ -1,297 +0,0 @@
/*
Unix SMB/CIFS implementation.
Infrastructure for async requests
Copyright (C) Volker Lendecke 2008
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
#include "includes.h"
#include "lib/tevent/tevent.h"
#include "lib/talloc/talloc.h"
#include "lib/util/dlinklist.h"
#include "lib/async_req/async_req.h"
#ifndef TALLOC_FREE
#define TALLOC_FREE(ctx) do { talloc_free(ctx); ctx=NULL; } while(0)
#endif
/**
* @brief Print an async_req structure
* @param[in] mem_ctx The memory context for the result
* @param[in] req The request to be printed
* @retval Text representation of req
*
* This is a default print function for async requests. Implementations should
* override this with more specific information.
*
* This function should not be used by async API users, this is non-static
* only to allow implementations to easily provide default information in
* their specific functions.
*/
char *async_req_print(TALLOC_CTX *mem_ctx, struct async_req *req)
{
return talloc_asprintf(mem_ctx, "async_req: state=%d, error=%d, "
"priv=%s", req->state, (int)req->error,
talloc_get_name(req->private_data));
}
/**
* @brief Create an async request
* @param[in] mem_ctx The memory context for the result
* @param[in] ev The event context this async request will be driven by
* @retval A new async request
*
* The new async request will be initialized in state ASYNC_REQ_IN_PROGRESS
*/
struct async_req *async_req_new(TALLOC_CTX *mem_ctx)
{
struct async_req *result;
result = talloc_zero(mem_ctx, struct async_req);
if (result == NULL) {
return NULL;
}
result->state = ASYNC_REQ_IN_PROGRESS;
result->print = async_req_print;
return result;
}
static void async_req_finish(struct async_req *req, enum async_req_state state)
{
req->state = state;
if (req->async.fn != NULL) {
req->async.fn(req);
}
}
/**
* @brief An async request has successfully finished
* @param[in] req The finished request
*
* async_req_done is to be used by implementors of async requests. When a
* request is successfully finished, this function calls the user's completion
* function.
*/
void async_req_done(struct async_req *req)
{
async_req_finish(req, ASYNC_REQ_DONE);
}
/**
* @brief An async request has seen an error
* @param[in] req The request with an error
* @param[in] error The error code
*
* async_req_done is to be used by implementors of async requests. When a
* request can not successfully completed, the implementation should call this
* function with the appropriate status code.
*/
void async_req_error(struct async_req *req, uint64_t error)
{
req->error = error;
async_req_finish(req, ASYNC_REQ_USER_ERROR);
}
/**
* @brief Timed event callback
* @param[in] ev Event context
* @param[in] te The timed event
* @param[in] now zero time
* @param[in] priv The async request to be finished
*/
static void async_trigger(struct tevent_context *ev, struct tevent_timer *te,
struct timeval now, void *priv)
{
struct async_req *req = talloc_get_type_abort(priv, struct async_req);
TALLOC_FREE(te);
if (req->error == 0) {
async_req_done(req);
}
else {
async_req_error(req, req->error);
}
}
/**
* @brief Helper function for nomem check
* @param[in] p The pointer to be checked
* @param[in] req The request being processed
*
* Convenience helper to easily check alloc failure within a callback
* implementing the next step of an async request.
*
* Call pattern would be
* \code
* p = talloc(mem_ctx, bla);
* if (async_req_ntnomem(p, req)) {
* return;
* }
* \endcode
*/
bool async_req_nomem(const void *p, struct async_req *req)
{
if (p != NULL) {
return false;
}
async_req_finish(req, ASYNC_REQ_NO_MEMORY);
return true;
}
/**
* @brief Finish a request before it started processing
* @param[in] req The finished request
* @param[in] status The success code
*
* An implementation of an async request might find that it can either finish
* the request without waiting for an external event, or it can't even start
* the engine. To present the illusion of a callback to the user of the API,
* the implementation can call this helper function which triggers an
* immediate timed event. This way the caller can use the same calling
* conventions, independent of whether the request was actually deferred.
*/
bool async_post_error(struct async_req *req, struct tevent_context *ev,
uint64_t error)
{
req->error = error;
if (tevent_add_timer(ev, req, tevent_timeval_zero(),
async_trigger, req) == NULL) {
return false;
}
return true;
}
bool async_req_is_error(struct async_req *req, enum async_req_state *state,
uint64_t *error)
{
if (req->state == ASYNC_REQ_DONE) {
return false;
}
if (req->state == ASYNC_REQ_USER_ERROR) {
*error = req->error;
}
*state = req->state;
return true;
}
struct async_queue_entry {
struct async_queue_entry *prev, *next;
struct async_req_queue *queue;
struct async_req *req;
void (*trigger)(struct async_req *req);
};
struct async_req_queue {
struct async_queue_entry *queue;
};
struct async_req_queue *async_req_queue_init(TALLOC_CTX *mem_ctx)
{
return talloc_zero(mem_ctx, struct async_req_queue);
}
static int async_queue_entry_destructor(struct async_queue_entry *e)
{
struct async_req_queue *queue = e->queue;
DLIST_REMOVE(queue->queue, e);
if (queue->queue != NULL) {
queue->queue->trigger(queue->queue->req);
}
return 0;
}
static void async_req_immediate_trigger(struct tevent_context *ev,
struct tevent_timer *te,
struct timeval now,
void *priv)
{
struct async_queue_entry *e = talloc_get_type_abort(
priv, struct async_queue_entry);
TALLOC_FREE(te);
e->trigger(e->req);
}
bool async_req_enqueue(struct async_req_queue *queue, struct tevent_context *ev,
struct async_req *req,
void (*trigger)(struct async_req *req))
{
struct async_queue_entry *e;
bool busy;
busy = (queue->queue != NULL);
e = talloc(req, struct async_queue_entry);
if (e == NULL) {
return false;
}
e->req = req;
e->trigger = trigger;
e->queue = queue;
DLIST_ADD_END(queue->queue, e, struct async_queue_entry *);
talloc_set_destructor(e, async_queue_entry_destructor);
if (!busy) {
struct tevent_timer *te;
te = tevent_add_timer(ev, e, tevent_timeval_zero(),
async_req_immediate_trigger,
e);
if (te == NULL) {
TALLOC_FREE(e);
return false;
}
}
return true;
}
bool _async_req_setup(TALLOC_CTX *mem_ctx, struct async_req **preq,
void *pstate, size_t state_size, const char *typename)
{
struct async_req *req;
void **ppstate = (void **)pstate;
void *state;
req = async_req_new(mem_ctx);
if (req == NULL) {
return false;
}
state = talloc_size(req, state_size);
if (state == NULL) {
TALLOC_FREE(req);
return false;
}
talloc_set_name_const(state, typename);
req->private_data = state;
*preq = req;
*ppstate = state;
return true;
}

View File

@ -1,158 +0,0 @@
/*
Unix SMB/CIFS implementation.
Infrastructure for async requests
Copyright (C) Volker Lendecke 2008
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
#ifndef __ASYNC_REQ_H__
#define __ASYNC_REQ_H__
#include "lib/talloc/talloc.h"
/**
* An async request moves between the following 4 states:
*/
enum async_req_state {
/**
* we are creating the request
*/
ASYNC_REQ_INIT,
/**
* we are waiting the request to complete
*/
ASYNC_REQ_IN_PROGRESS,
/**
* the request is finished
*/
ASYNC_REQ_DONE,
/**
* A user error has occured
*/
ASYNC_REQ_USER_ERROR,
/**
* Request timed out
*/
ASYNC_REQ_TIMED_OUT,
/**
* No memory in between
*/
ASYNC_REQ_NO_MEMORY
};
/**
* @brief An async request
*
* This represents an async request being processed by callbacks via an event
* context. A user can issue for example a write request to a socket, giving
* an implementation function the fd, the buffer and the number of bytes to
* transfer. The function issuing the request will immediately return without
* blocking most likely without having sent anything. The API user then fills
* in req->async.fn and req->async.priv, functions that are called when the
* request is finished.
*
* It is up to the user of the async request to talloc_free it after it has
* finished. This can happen while the completion function is called.
*/
struct async_req {
/**
* @brief The external state - will be queried by the caller
*
* While the async request is being processed, state will remain in
* ASYNC_REQ_IN_PROGRESS. A request is finished if
* req->state>=ASYNC_REQ_DONE.
*/
enum async_req_state state;
/**
* @brief Private pointer for the actual implementation
*
* The implementation doing the work for the async request needs a
* current state like for example a fd event. The user of an async
* request should not touch this.
*/
void *private_data;
/**
* @brief Print yourself, for debugging purposes
*
* Async requests are opaque data structures. The implementation of an
* async request can define a custom function to print more debug
* info.
*/
char *(*print)(TALLOC_CTX *mem_ctx, struct async_req *);
/**
* @brief status code when finished
*
* This status can be queried in the async completion function. It
* will be set to 0 when everything went fine.
**/
uint64_t error;
/**
* @brief What to do on completion
*
* This is used for the user of an async request, fn is called when
* the request completes, either successfully or with an error.
*/
struct {
/**
* @brief Completion function
* Completion function, to be filled by the API user
*/
void (*fn)(struct async_req *);
/**
* @brief Private data for the completion function
*/
void *priv;
} async;
};
struct async_req *async_req_new(TALLOC_CTX *mem_ctx);
char *async_req_print(TALLOC_CTX *mem_ctx, struct async_req *req);
void async_req_done(struct async_req *req);
void async_req_error(struct async_req *req, uint64_t error);
bool async_req_nomem(const void *p, struct async_req *req);
bool async_post_error(struct async_req *req, struct tevent_context *ev,
uint64_t error);
bool async_req_is_error(struct async_req *req, enum async_req_state *state,
uint64_t *error);
struct async_req_queue;
struct async_req_queue *async_req_queue_init(TALLOC_CTX *mem_ctx);
bool async_req_enqueue(struct async_req_queue *queue,
struct tevent_context *ev,
struct async_req *req,
void (*trigger)(struct async_req *req));
bool _async_req_setup(TALLOC_CTX *mem_ctx, struct async_req **preq,
void *pstate, size_t state_size, const char *typename);
#define async_req_setup(_mem_ctx, _preq, _pstate, type) \
_async_req_setup((_mem_ctx), (_preq), (_pstate), sizeof(type), #type)
#endif

View File

@ -1,70 +0,0 @@
/*
Unix SMB/CIFS implementation.
NTSTATUS wrappers for async_req.h
Copyright (C) Volker Lendecke 2008, 2009
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
#include "includes.h"
#include "lib/tevent/tevent.h"
#include "lib/talloc/talloc.h"
#include "lib/util/dlinklist.h"
#include "lib/async_req/async_req_ntstatus.h"
void async_req_nterror(struct async_req *req, NTSTATUS status)
{
async_req_error(req, NT_STATUS_V(status));
}
bool async_post_ntstatus(struct async_req *req, struct tevent_context *ev,
NTSTATUS status)
{
return async_post_error(req, ev, NT_STATUS_V(status));
}
bool async_req_is_nterror(struct async_req *req, NTSTATUS *status)
{
enum async_req_state state;
uint64_t error;
if (!async_req_is_error(req, &state, &error)) {
return false;
}
switch (state) {
case ASYNC_REQ_USER_ERROR:
*status = NT_STATUS(error);
break;
case ASYNC_REQ_TIMED_OUT:
*status = NT_STATUS_IO_TIMEOUT;
break;
case ASYNC_REQ_NO_MEMORY:
*status = NT_STATUS_NO_MEMORY;
break;
default:
*status = NT_STATUS_INTERNAL_ERROR;
break;
}
return true;
}
NTSTATUS async_req_simple_recv_ntstatus(struct async_req *req)
{
NTSTATUS status;
if (async_req_is_nterror(req, &status)) {
return status;
}
return NT_STATUS_OK;
}

View File

@ -1,35 +0,0 @@
/*
Unix SMB/CIFS implementation.
NTSTATUS wrappers for async_req.h
Copyright (C) Volker Lendecke 2008, 2009
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
#ifndef __ASYNC_REQ_NTSTATUS_H__
#define __ASYNC_REQ_NTSTATUS_H__
#include "lib/async_req/async_req.h"
#include "includes.h"
void async_req_nterror(struct async_req *req, NTSTATUS status);
bool async_post_ntstatus(struct async_req *req, struct tevent_context *ev,
NTSTATUS status);
bool async_req_is_nterror(struct async_req *req, NTSTATUS *status);
NTSTATUS async_req_simple_recv_ntstatus(struct async_req *req);
#endif

View File

@ -20,7 +20,6 @@
#include "includes.h"
#include "lib/talloc/talloc.h"
#include "lib/tevent/tevent.h"
#include "lib/async_req/async_req.h"
#include "lib/async_req/async_sock.h"
#include "lib/util/tevent_unix.h"
#include <fcntl.h>
@ -29,55 +28,6 @@
#define TALLOC_FREE(ctx) do { talloc_free(ctx); ctx=NULL; } while(0)
#endif
/**
* @brief Map async_req states to unix-style errnos
* @param[in] req The async req to get the state from
* @param[out] err Pointer to take the unix-style errno
*
* @return true if the async_req is in an error state, false otherwise
*/
bool async_req_is_errno(struct async_req *req, int *err)
{
enum async_req_state state;
uint64_t error;
if (!async_req_is_error(req, &state, &error)) {
return false;
}
switch (state) {
case ASYNC_REQ_USER_ERROR:
*err = (int)error;
break;
case ASYNC_REQ_TIMED_OUT:
#ifdef ETIMEDOUT
*err = ETIMEDOUT;
#else
*err = EAGAIN;
#endif
break;
case ASYNC_REQ_NO_MEMORY:
*err = ENOMEM;
break;
default:
*err = EIO;
break;
}
return true;
}
int async_req_simple_recv_errno(struct async_req *req)
{
int err;
if (async_req_is_errno(req, &err)) {
return err;
}
return 0;
}
struct async_send_state {
int fd;
const void *buf;
@ -547,7 +497,8 @@ static void read_packet_handler(struct tevent_context *ev,
ssize_t nread, more;
uint8_t *tmp;
nread = read(state->fd, state->buf+state->nread, total-state->nread);
nread = recv(state->fd, state->buf+state->nread, total-state->nread,
0);
if (nread == -1) {
tevent_req_error(req, errno);
return;

View File

@ -22,9 +22,6 @@
#include "includes.h"
bool async_req_is_errno(struct async_req *req, int *err);
int async_req_simple_recv_errno(struct async_req *req);
struct tevent_req *async_send_send(TALLOC_CTX *mem_ctx,
struct tevent_context *ev,
int fd, const void *buf, size_t len,

View File

@ -1,3 +1,3 @@
[SUBSYSTEM::LIBASYNC_REQ]
LIBASYNC_REQ_OBJ_FILES = $(addprefix ../lib/async_req/, async_req.o async_sock.o async_req_ntstatus.o)
LIBASYNC_REQ_OBJ_FILES = $(addprefix ../lib/async_req/, async_sock.o)

View File

@ -83,7 +83,7 @@
#define POPT_ERROR_NOARG -10 /*!< missing argument */
#define POPT_ERROR_BADOPT -11 /*!< unknown option */
#define POPT_ERROR_OPTSTOODEEP -13 /*!< aliases nested too deeply */
#define POPT_ERROR_BADQUOTE -15 /*!< error in paramter quoting */
#define POPT_ERROR_BADQUOTE -15 /*!< error in parameter quoting */
#define POPT_ERROR_ERRNO -16 /*!< errno set, use strerror(errno) */
#define POPT_ERROR_BADNUMBER -17 /*!< invalid numeric value */
#define POPT_ERROR_OVERFLOW -18 /*!< number too large or too small */

View File

@ -5,8 +5,6 @@ LIBTSOCKET_OBJ_FILES = $(addprefix ../lib/tsocket/, \
tsocket.o \
tsocket_helpers.o \
tsocket_bsd.o \
tsocket_recvfrom.o \
tsocket_sendto.o \
tsocket_connect.o \
tsocket_writev.o \
tsocket_readv.o)

View File

@ -118,21 +118,6 @@ int tsocket_writev(struct tsocket_context *sock,
return sock->ops->writev_data(sock, vector, count);
}
ssize_t tsocket_recvfrom(struct tsocket_context *sock,
uint8_t *data, size_t len,
TALLOC_CTX *addr_ctx,
struct tsocket_address **src_addr)
{
return sock->ops->recvfrom_data(sock, data, len, addr_ctx, src_addr);
}
ssize_t tsocket_sendto(struct tsocket_context *sock,
const uint8_t *data, size_t len,
const struct tsocket_address *dest_addr)
{
return sock->ops->sendto_data(sock, data, len, dest_addr);
}
int tsocket_get_status(const struct tsocket_context *sock)
{
return sock->ops->get_status(sock);
@ -229,3 +214,397 @@ int _tsocket_address_create_socket(const struct tsocket_address *addr,
return addr->ops->create_socket(addr, type, mem_ctx, sock, location);
}
struct tdgram_context {
const char *location;
const struct tdgram_context_ops *ops;
void *private_data;
};
struct tdgram_context *_tdgram_context_create(TALLOC_CTX *mem_ctx,
const struct tdgram_context_ops *ops,
void *pstate,
size_t psize,
const char *type,
const char *location)
{
struct tdgram_context *dgram;
void **ppstate = (void **)pstate;
void *state;
dgram = talloc(mem_ctx, struct tdgram_context);
if (dgram == NULL) {
return NULL;
}
dgram->location = location;
dgram->ops = ops;
state = talloc_size(dgram, psize);
if (state == NULL) {
talloc_free(dgram);
return NULL;
}
talloc_set_name_const(state, type);
dgram->private_data = state;
*ppstate = state;
return dgram;
}
void *_tdgram_context_data(struct tdgram_context *dgram)
{
return dgram->private_data;
}
struct tdgram_recvfrom_state {
const struct tdgram_context_ops *ops;
uint8_t *buf;
size_t len;
struct tsocket_address *src;
};
static void tdgram_recvfrom_done(struct tevent_req *subreq);
struct tevent_req *tdgram_recvfrom_send(TALLOC_CTX *mem_ctx,
struct tevent_context *ev,
struct tdgram_context *dgram)
{
struct tevent_req *req;
struct tdgram_recvfrom_state *state;
struct tevent_req *subreq;
req = tevent_req_create(mem_ctx, &state,
struct tdgram_recvfrom_state);
if (req == NULL) {
return NULL;
}
state->ops = dgram->ops;
subreq = state->ops->recvfrom_send(state, ev, dgram);
if (tevent_req_nomem(subreq, req)) {
goto post;
}
tevent_req_set_callback(subreq, tdgram_recvfrom_done, req);
return req;
post:
tevent_req_post(req, ev);
return req;
}
static void tdgram_recvfrom_done(struct tevent_req *subreq)
{
struct tevent_req *req = tevent_req_callback_data(subreq,
struct tevent_req);
struct tdgram_recvfrom_state *state = tevent_req_data(req,
struct tdgram_recvfrom_state);
ssize_t ret;
int sys_errno;
ret = state->ops->recvfrom_recv(subreq, &sys_errno, state,
&state->buf, &state->src);
if (ret == -1) {
tevent_req_error(req, sys_errno);
return;
}
state->len = ret;
tevent_req_done(req);
}
ssize_t tdgram_recvfrom_recv(struct tevent_req *req,
int *perrno,
TALLOC_CTX *mem_ctx,
uint8_t **buf,
struct tsocket_address **src)
{
struct tdgram_recvfrom_state *state = tevent_req_data(req,
struct tdgram_recvfrom_state);
ssize_t ret;
ret = tsocket_simple_int_recv(req, perrno);
if (ret == 0) {
*buf = talloc_move(mem_ctx, &state->buf);
ret = state->len;
if (src) {
*src = talloc_move(mem_ctx, &state->src);
}
}
tevent_req_received(req);
return ret;
}
struct tdgram_sendto_state {
const struct tdgram_context_ops *ops;
ssize_t ret;
};
static void tdgram_sendto_done(struct tevent_req *subreq);
struct tevent_req *tdgram_sendto_send(TALLOC_CTX *mem_ctx,
struct tevent_context *ev,
struct tdgram_context *dgram,
const uint8_t *buf, size_t len,
const struct tsocket_address *dst)
{
struct tevent_req *req;
struct tdgram_sendto_state *state;
struct tevent_req *subreq;
req = tevent_req_create(mem_ctx, &state,
struct tdgram_sendto_state);
if (req == NULL) {
return NULL;
}
state->ops = dgram->ops;
state->ret = -1;
subreq = state->ops->sendto_send(state, ev, dgram,
buf, len, dst);
if (tevent_req_nomem(subreq, req)) {
goto post;
}
tevent_req_set_callback(subreq, tdgram_sendto_done, req);
return req;
post:
tevent_req_post(req, ev);
return req;
}
static void tdgram_sendto_done(struct tevent_req *subreq)
{
struct tevent_req *req = tevent_req_callback_data(subreq,
struct tevent_req);
struct tdgram_sendto_state *state = tevent_req_data(req,
struct tdgram_sendto_state);
ssize_t ret;
int sys_errno;
ret = state->ops->sendto_recv(subreq, &sys_errno);
if (ret == -1) {
tevent_req_error(req, sys_errno);
return;
}
state->ret = ret;
tevent_req_done(req);
}
ssize_t tdgram_sendto_recv(struct tevent_req *req,
int *perrno)
{
struct tdgram_sendto_state *state = tevent_req_data(req,
struct tdgram_sendto_state);
ssize_t ret;
ret = tsocket_simple_int_recv(req, perrno);
if (ret == 0) {
ret = state->ret;
}
tevent_req_received(req);
return ret;
}
struct tdgram_disconnect_state {
const struct tdgram_context_ops *ops;
};
static void tdgram_disconnect_done(struct tevent_req *subreq);
struct tevent_req *tdgram_disconnect_send(TALLOC_CTX *mem_ctx,
struct tevent_context *ev,
struct tdgram_context *dgram)
{
struct tevent_req *req;
struct tdgram_disconnect_state *state;
struct tevent_req *subreq;
req = tevent_req_create(mem_ctx, &state,
struct tdgram_disconnect_state);
if (req == NULL) {
return NULL;
}
state->ops = dgram->ops;
subreq = state->ops->disconnect_send(state, ev, dgram);
if (tevent_req_nomem(subreq, req)) {
goto post;
}
tevent_req_set_callback(subreq, tdgram_disconnect_done, req);
return req;
post:
tevent_req_post(req, ev);
return req;
}
static void tdgram_disconnect_done(struct tevent_req *subreq)
{
struct tevent_req *req = tevent_req_callback_data(subreq,
struct tevent_req);
struct tdgram_disconnect_state *state = tevent_req_data(req,
struct tdgram_disconnect_state);
int ret;
int sys_errno;
ret = state->ops->disconnect_recv(subreq, &sys_errno);
if (ret == -1) {
tevent_req_error(req, sys_errno);
return;
}
tevent_req_done(req);
}
int tdgram_disconnect_recv(struct tevent_req *req,
int *perrno)
{
int ret;
ret = tsocket_simple_int_recv(req, perrno);
tevent_req_received(req);
return ret;
}
struct tdgram_sendto_queue_state {
/* this structs are owned by the caller */
struct {
struct tevent_context *ev;
struct tdgram_context *dgram;
const uint8_t *buf;
size_t len;
const struct tsocket_address *dst;
} caller;
ssize_t ret;
};
static void tdgram_sendto_queue_trigger(struct tevent_req *req,
void *private_data);
static void tdgram_sendto_queue_done(struct tevent_req *subreq);
/**
* @brief Queue a dgram blob for sending through the socket
* @param[in] mem_ctx The memory context for the result
* @param[in] ev The event context the operation should work on
* @param[in] dgram The tdgram_context to send the message buffer
* @param[in] queue The existing dgram queue
* @param[in] buf The message buffer
* @param[in] len The message length
* @param[in] dst The destination socket address
* @retval The async request handle
*
* This function queues a blob for sending to destination through an existing
* dgram socket. The async callback is triggered when the whole blob is
* delivered to the underlying system socket.
*
* The caller needs to make sure that all non-scalar input parameters hang
* arround for the whole lifetime of the request.
*/
struct tevent_req *tdgram_sendto_queue_send(TALLOC_CTX *mem_ctx,
struct tevent_context *ev,
struct tdgram_context *dgram,
struct tevent_queue *queue,
const uint8_t *buf,
size_t len,
struct tsocket_address *dst)
{
struct tevent_req *req;
struct tdgram_sendto_queue_state *state;
bool ok;
req = tevent_req_create(mem_ctx, &state,
struct tdgram_sendto_queue_state);
if (!req) {
return NULL;
}
state->caller.ev = ev;
state->caller.dgram = dgram;
state->caller.buf = buf;
state->caller.len = len;
state->caller.dst = dst;
state->ret = -1;
ok = tevent_queue_add(queue,
ev,
req,
tdgram_sendto_queue_trigger,
NULL);
if (!ok) {
tevent_req_nomem(NULL, req);
goto post;
}
return req;
post:
tevent_req_post(req, ev);
return req;
}
static void tdgram_sendto_queue_trigger(struct tevent_req *req,
void *private_data)
{
struct tdgram_sendto_queue_state *state = tevent_req_data(req,
struct tdgram_sendto_queue_state);
struct tevent_req *subreq;
subreq = tdgram_sendto_send(state,
state->caller.ev,
state->caller.dgram,
state->caller.buf,
state->caller.len,
state->caller.dst);
if (tevent_req_nomem(subreq, req)) {
return;
}
tevent_req_set_callback(subreq, tdgram_sendto_queue_done, req);
}
static void tdgram_sendto_queue_done(struct tevent_req *subreq)
{
struct tevent_req *req = tevent_req_callback_data(subreq,
struct tevent_req);
struct tdgram_sendto_queue_state *state = tevent_req_data(req,
struct tdgram_sendto_queue_state);
ssize_t ret;
int sys_errno;
ret = tdgram_sendto_recv(subreq, &sys_errno);
talloc_free(subreq);
if (ret == -1) {
tevent_req_error(req, sys_errno);
return;
}
state->ret = ret;
tevent_req_done(req);
}
ssize_t tdgram_sendto_queue_recv(struct tevent_req *req, int *perrno)
{
struct tdgram_sendto_queue_state *state = tevent_req_data(req,
struct tdgram_sendto_queue_state);
ssize_t ret;
ret = tsocket_simple_int_recv(req, perrno);
if (ret == 0) {
ret = state->ret;
}
tevent_req_received(req);
return ret;
}

View File

@ -29,11 +29,11 @@
struct tsocket_context;
struct tsocket_address;
struct tdgram_context;
struct iovec;
enum tsocket_type {
TSOCKET_TYPE_STREAM = 1,
TSOCKET_TYPE_DGRAM,
TSOCKET_TYPE_MESSAGE
};
@ -67,14 +67,6 @@ int tsocket_readv(struct tsocket_context *sock,
int tsocket_writev(struct tsocket_context *sock,
const struct iovec *vector, size_t count);
ssize_t tsocket_recvfrom(struct tsocket_context *sock,
uint8_t *data, size_t len,
TALLOC_CTX *addr_ctx,
struct tsocket_address **src_addr);
ssize_t tsocket_sendto(struct tsocket_context *sock,
const uint8_t *data, size_t len,
const struct tsocket_address *dest_addr);
int tsocket_get_status(const struct tsocket_context *sock);
int _tsocket_get_local_address(const struct tsocket_context *sock,
@ -120,6 +112,32 @@ int _tsocket_address_create_socket(const struct tsocket_address *addr,
_tsocket_address_create_socket(addr, type, mem_ctx, sock,\
__location__)
/*
* tdgram_context related functions
*/
struct tevent_req *tdgram_recvfrom_send(TALLOC_CTX *mem_ctx,
struct tevent_context *ev,
struct tdgram_context *dgram);
ssize_t tdgram_recvfrom_recv(struct tevent_req *req,
int *perrno,
TALLOC_CTX *mem_ctx,
uint8_t **buf,
struct tsocket_address **src);
struct tevent_req *tdgram_sendto_send(TALLOC_CTX *mem_ctx,
struct tevent_context *ev,
struct tdgram_context *dgram,
const uint8_t *buf, size_t len,
const struct tsocket_address *dst);
ssize_t tdgram_sendto_recv(struct tevent_req *req,
int *perrno);
struct tevent_req *tdgram_disconnect_send(TALLOC_CTX *mem_ctx,
struct tevent_context *ev,
struct tdgram_context *dgram);
int tdgram_disconnect_recv(struct tevent_req *req,
int *perrno);
/*
* BSD sockets: inet, inet6 and unix
*/
@ -160,33 +178,26 @@ int _tsocket_context_bsd_wrap_existing(TALLOC_CTX *mem_ctx,
_tsocket_context_bsd_wrap_existing(mem_ctx, fd, cod, _sock, \
__location__)
int _tdgram_inet_udp_socket(const struct tsocket_address *local,
const struct tsocket_address *remote,
TALLOC_CTX *mem_ctx,
struct tdgram_context **dgram,
const char *location);
#define tdgram_inet_udp_socket(local, remote, mem_ctx, dgram) \
_tdgram_inet_udp_socket(local, remote, mem_ctx, dgram, __location__)
int _tdgram_unix_dgram_socket(const struct tsocket_address *local,
const struct tsocket_address *remote,
TALLOC_CTX *mem_ctx,
struct tdgram_context **dgram,
const char *location);
#define tdgram_unix_dgram_socket(local, remote, mem_ctx, dgram) \
_tdgram_unix_dgram_socket(local, remote, mem_ctx, dgram, __location__)
/*
* Async helpers
*/
struct tevent_req *tsocket_recvfrom_send(struct tsocket_context *sock,
TALLOC_CTX *mem_ctx);
ssize_t tsocket_recvfrom_recv(struct tevent_req *req,
int *perrno,
TALLOC_CTX *mem_ctx,
uint8_t **buf,
struct tsocket_address **src);
struct tevent_req *tsocket_sendto_send(struct tsocket_context *sock,
TALLOC_CTX *mem_ctx,
const uint8_t *buf,
size_t len,
const struct tsocket_address *dst);
ssize_t tsocket_sendto_recv(struct tevent_req *req, int *perrno);
struct tevent_req *tsocket_sendto_queue_send(TALLOC_CTX *mem_ctx,
struct tsocket_context *sock,
struct tevent_queue *queue,
const uint8_t *buf,
size_t len,
struct tsocket_address *dst);
ssize_t tsocket_sendto_queue_recv(struct tevent_req *req, int *perrno);
struct tevent_req *tsocket_connect_send(struct tsocket_context *sock,
TALLOC_CTX *mem_ctx,
const struct tsocket_address *dst);
@ -216,5 +227,18 @@ struct tevent_req *tsocket_readv_send(struct tsocket_context *sock,
void *private_data);
int tsocket_readv_recv(struct tevent_req *req, int *perrno);
/*
* Queue helpers
*/
struct tevent_req *tdgram_sendto_queue_send(TALLOC_CTX *mem_ctx,
struct tevent_context *ev,
struct tdgram_context *dgram,
struct tevent_queue *queue,
const uint8_t *buf,
size_t len,
struct tsocket_address *dst);
ssize_t tdgram_sendto_queue_recv(struct tevent_req *req, int *perrno);
#endif /* _TSOCKET_H */

View File

@ -24,9 +24,168 @@
#include "replace.h"
#include "system/filesys.h"
#include "system/network.h"
#include "system/filesys.h"
#include "tsocket.h"
#include "tsocket_internal.h"
static int tsocket_bsd_error_from_errno(int ret,
int sys_errno,
bool *retry)
{
*retry = false;
if (ret >= 0) {
return 0;
}
if (ret != -1) {
return EIO;
}
if (sys_errno == 0) {
return EIO;
}
if (sys_errno == EINTR) {
*retry = true;
return sys_errno;
}
if (sys_errno == EINPROGRESS) {
*retry = true;
return sys_errno;
}
if (sys_errno == EAGAIN) {
*retry = true;
return sys_errno;
}
#ifdef EWOULDBLOCK
if (sys_errno == EWOULDBLOCK) {
*retry = true;
return sys_errno;
}
#endif
return sys_errno;
}
static int tsocket_bsd_common_prepare_fd(int fd, bool high_fd)
{
int i;
int sys_errno = 0;
int fds[3];
int num_fds = 0;
int result, flags;
if (fd == -1) {
return -1;
}
/* first make a fd >= 3 */
if (high_fd) {
while (fd < 3) {
fds[num_fds++] = fd;
fd = dup(fd);
if (fd == -1) {
sys_errno = errno;
break;
}
}
for (i=0; i<num_fds; i++) {
close(fds[i]);
}
if (fd == -1) {
errno = sys_errno;
return fd;
}
}
/* fd should be nonblocking. */
#ifdef O_NONBLOCK
#define FLAG_TO_SET O_NONBLOCK
#else
#ifdef SYSV
#define FLAG_TO_SET O_NDELAY
#else /* BSD */
#define FLAG_TO_SET FNDELAY
#endif
#endif
if ((flags = fcntl(fd, F_GETFL)) == -1) {
goto fail;
}
flags |= FLAG_TO_SET;
if (fcntl(fd, F_SETFL, flags) == -1) {
goto fail;
}
#undef FLAG_TO_SET
/* fd should be closed on exec() */
#ifdef FD_CLOEXEC
result = flags = fcntl(fd, F_GETFD, 0);
if (flags >= 0) {
flags |= FD_CLOEXEC;
result = fcntl(fd, F_SETFD, flags);
}
if (result < 0) {
goto fail;
}
#endif
return fd;
fail:
if (fd != -1) {
sys_errno = errno;
close(fd);
errno = sys_errno;
}
return -1;
}
static ssize_t tsocket_bsd_pending(int fd)
{
int ret;
int value = 0;
ret = ioctl(fd, FIONREAD, &value);
if (ret == -1) {
return ret;
}
if (ret == 0) {
if (value == 0) {
int error=0;
socklen_t len = sizeof(error);
/*
* if no data is available check if the socket
* is in error state. For dgram sockets
* it's the way to return ICMP error messages
* of connected sockets to the caller.
*/
ret = getsockopt(fd, SOL_SOCKET, SO_ERROR,
&error, &len);
if (ret == -1) {
return ret;
}
if (error != 0) {
errno = error;
return -1;
}
}
return value;
}
/* this should not be reached */
errno = EIO;
return -1;
}
static const struct tsocket_context_ops tsocket_context_bsd_ops;
static const struct tsocket_address_ops tsocket_address_bsd_ops;
@ -456,9 +615,6 @@ static int tsocket_address_bsd_create_socket(const struct tsocket_address *addr,
}
bsd_type = SOCK_STREAM;
break;
case TSOCKET_TYPE_DGRAM:
bsd_type = SOCK_DGRAM;
break;
default:
errno = EPROTONOSUPPORT;
return -1;
@ -785,73 +941,6 @@ static int tsocket_context_bsd_writev_data(struct tsocket_context *sock,
return ret;
}
static ssize_t tsocket_context_bsd_recvfrom_data(struct tsocket_context *sock,
uint8_t *data, size_t len,
TALLOC_CTX *addr_ctx,
struct tsocket_address **remote)
{
struct tsocket_context_bsd *bsds = talloc_get_type(sock->private_data,
struct tsocket_context_bsd);
struct tsocket_address *addr = NULL;
struct tsocket_address_bsd *bsda;
ssize_t ret;
struct sockaddr *sa = NULL;
socklen_t sa_len = 0;
if (remote) {
addr = tsocket_address_create(addr_ctx,
&tsocket_address_bsd_ops,
&bsda,
struct tsocket_address_bsd,
__location__ "recvfrom");
if (!addr) {
return -1;
}
ZERO_STRUCTP(bsda);
sa = &bsda->u.sa;
sa_len = sizeof(bsda->u.ss);
}
ret = recvfrom(bsds->fd, data, len, 0, sa, &sa_len);
if (ret < 0) {
int saved_errno = errno;
talloc_free(addr);
errno = saved_errno;
return ret;
}
if (remote) {
*remote = addr;
}
return ret;
}
static ssize_t tsocket_context_bsd_sendto_data(struct tsocket_context *sock,
const uint8_t *data, size_t len,
const struct tsocket_address *remote)
{
struct tsocket_context_bsd *bsds = talloc_get_type(sock->private_data,
struct tsocket_context_bsd);
struct sockaddr *sa = NULL;
socklen_t sa_len = 0;
ssize_t ret;
if (remote) {
struct tsocket_address_bsd *bsda =
talloc_get_type(remote->private_data,
struct tsocket_address_bsd);
sa = &bsda->u.sa;
sa_len = sizeof(bsda->u.ss);
}
ret = sendto(bsds->fd, data, len, 0, sa, sa_len);
return ret;
}
static int tsocket_context_bsd_get_status(const struct tsocket_context *sock)
{
struct tsocket_context_bsd *bsds = talloc_get_type(sock->private_data,
@ -1113,8 +1202,6 @@ static const struct tsocket_context_ops tsocket_context_bsd_ops = {
.pending_data = tsocket_context_bsd_pending_data,
.readv_data = tsocket_context_bsd_readv_data,
.writev_data = tsocket_context_bsd_writev_data,
.recvfrom_data = tsocket_context_bsd_recvfrom_data,
.sendto_data = tsocket_context_bsd_sendto_data,
.get_status = tsocket_context_bsd_get_status,
.get_local_address = tsocket_context_bsd_get_local_address,
@ -1125,3 +1212,709 @@ static const struct tsocket_context_ops tsocket_context_bsd_ops = {
.disconnect = tsocket_context_bsd_disconnect
};
struct tdgram_bsd {
int fd;
void *event_ptr;
struct tevent_fd *fde;
void *readable_private;
void (*readable_handler)(void *private_data);
void *writeable_private;
void (*writeable_handler)(void *private_data);
struct tevent_req *read_req;
struct tevent_req *write_req;
};
static void tdgram_bsd_fde_handler(struct tevent_context *ev,
struct tevent_fd *fde,
uint16_t flags,
void *private_data)
{
struct tdgram_bsd *bsds = talloc_get_type_abort(private_data,
struct tdgram_bsd);
if (flags & TEVENT_FD_WRITE) {
bsds->writeable_handler(bsds->writeable_private);
return;
}
if (flags & TEVENT_FD_READ) {
if (!bsds->readable_handler) {
TEVENT_FD_NOT_READABLE(bsds->fde);
return;
}
bsds->readable_handler(bsds->readable_private);
return;
}
}
static int tdgram_bsd_set_readable_handler(struct tdgram_bsd *bsds,
struct tevent_context *ev,
void (*handler)(void *private_data),
void *private_data)
{
if (ev == NULL) {
if (handler) {
errno = EINVAL;
return -1;
}
if (!bsds->readable_handler) {
return 0;
}
bsds->readable_handler = NULL;
bsds->readable_private = NULL;
return 0;
}
/* read and write must use the same tevent_context */
if (bsds->event_ptr != ev) {
if (bsds->readable_handler || bsds->writeable_handler) {
errno = EINVAL;
return -1;
}
bsds->event_ptr = NULL;
TALLOC_FREE(bsds->fde);
}
if (bsds->fde == NULL) {
bsds->fde = tevent_add_fd(ev, bsds,
bsds->fd, TEVENT_FD_READ,
tdgram_bsd_fde_handler,
bsds);
if (!bsds->fde) {
return -1;
}
/* cache the event context we're running on */
bsds->event_ptr = ev;
} else if (!bsds->readable_handler) {
TEVENT_FD_READABLE(bsds->fde);
}
bsds->readable_handler = handler;
bsds->readable_private = private_data;
return 0;
}
static int tdgram_bsd_set_writeable_handler(struct tdgram_bsd *bsds,
struct tevent_context *ev,
void (*handler)(void *private_data),
void *private_data)
{
if (ev == NULL) {
if (handler) {
errno = EINVAL;
return -1;
}
if (!bsds->writeable_handler) {
return 0;
}
bsds->writeable_handler = NULL;
bsds->writeable_private = NULL;
TEVENT_FD_NOT_WRITEABLE(bsds->fde);
return 0;
}
/* read and write must use the same tevent_context */
if (bsds->event_ptr != ev) {
if (bsds->readable_handler || bsds->writeable_handler) {
errno = EINVAL;
return -1;
}
bsds->event_ptr = NULL;
TALLOC_FREE(bsds->fde);
}
if (bsds->fde == NULL) {
bsds->fde = tevent_add_fd(ev, bsds,
bsds->fd, TEVENT_FD_WRITE,
tdgram_bsd_fde_handler,
bsds);
if (!bsds->fde) {
return -1;
}
/* cache the event context we're running on */
bsds->event_ptr = ev;
} else if (!bsds->writeable_handler) {
TEVENT_FD_WRITEABLE(bsds->fde);
}
bsds->writeable_handler = handler;
bsds->writeable_private = private_data;
return 0;
}
struct tdgram_bsd_recvfrom_state {
struct tdgram_context *dgram;
uint8_t *buf;
size_t len;
struct tsocket_address *src;
};
static int tdgram_bsd_recvfrom_destructor(struct tdgram_bsd_recvfrom_state *state)
{
struct tdgram_bsd *bsds = tdgram_context_data(state->dgram,
struct tdgram_bsd);
bsds->read_req = NULL;
tdgram_bsd_set_readable_handler(bsds, NULL, NULL, NULL);
return 0;
}
static void tdgram_bsd_recvfrom_handler(void *private_data);
static struct tevent_req *tdgram_bsd_recvfrom_send(TALLOC_CTX *mem_ctx,
struct tevent_context *ev,
struct tdgram_context *dgram)
{
struct tevent_req *req;
struct tdgram_bsd_recvfrom_state *state;
struct tdgram_bsd *bsds = tdgram_context_data(dgram, struct tdgram_bsd);
int ret;
req = tevent_req_create(mem_ctx, &state,
struct tdgram_bsd_recvfrom_state);
if (!req) {
return NULL;
}
state->dgram = dgram;
state->buf = NULL;
state->len = 0;
state->src = NULL;
if (bsds->read_req) {
tevent_req_error(req, EBUSY);
goto post;
}
bsds->read_req = req;
talloc_set_destructor(state, tdgram_bsd_recvfrom_destructor);
if (bsds->fd == -1) {
tevent_req_error(req, ENOTCONN);
goto post;
}
/*
* this is a fast path, not waiting for the
* socket to become explicit readable gains
* about 10%-20% performance in benchmark tests.
*/
tdgram_bsd_recvfrom_handler(req);
if (!tevent_req_is_in_progress(req)) {
goto post;
}
ret = tdgram_bsd_set_readable_handler(bsds, ev,
tdgram_bsd_recvfrom_handler,
req);
if (ret == -1) {
tevent_req_error(req, errno);
goto post;
}
return req;
post:
tevent_req_post(req, ev);
return req;
}
static void tdgram_bsd_recvfrom_handler(void *private_data)
{
struct tevent_req *req = talloc_get_type_abort(private_data,
struct tevent_req);
struct tdgram_bsd_recvfrom_state *state = tevent_req_data(req,
struct tdgram_bsd_recvfrom_state);
struct tdgram_context *dgram = state->dgram;
struct tdgram_bsd *bsds = tdgram_context_data(dgram, struct tdgram_bsd);
struct tsocket_address_bsd *bsda;
ssize_t ret;
struct sockaddr *sa = NULL;
socklen_t sa_len = 0;
int err;
bool retry;
ret = tsocket_bsd_pending(bsds->fd);
if (ret == 0) {
/* retry later */
return;
}
err = tsocket_bsd_error_from_errno(ret, errno, &retry);
if (retry) {
/* retry later */
return;
}
if (tevent_req_error(req, err)) {
return;
}
state->buf = talloc_array(state, uint8_t, ret);
if (tevent_req_nomem(state->buf, req)) {
return;
}
state->len = ret;
state->src = tsocket_address_create(state,
&tsocket_address_bsd_ops,
&bsda,
struct tsocket_address_bsd,
__location__ "bsd_recvfrom");
if (tevent_req_nomem(state->src, req)) {
return;
}
ZERO_STRUCTP(bsda);
sa = &bsda->u.sa;
sa_len = sizeof(bsda->u.ss);
ret = recvfrom(bsds->fd, state->buf, state->len, 0, sa, &sa_len);
err = tsocket_error_from_errno(ret, errno, &retry);
if (retry) {
/* retry later */
return;
}
if (tevent_req_error(req, err)) {
return;
}
if (ret != state->len) {
tevent_req_error(req, EIO);
return;
}
tevent_req_done(req);
}
static ssize_t tdgram_bsd_recvfrom_recv(struct tevent_req *req,
int *perrno,
TALLOC_CTX *mem_ctx,
uint8_t **buf,
struct tsocket_address **src)
{
struct tdgram_bsd_recvfrom_state *state = tevent_req_data(req,
struct tdgram_bsd_recvfrom_state);
ssize_t ret;
ret = tsocket_simple_int_recv(req, perrno);
if (ret == 0) {
*buf = talloc_move(mem_ctx, &state->buf);
ret = state->len;
if (src) {
*src = talloc_move(mem_ctx, &state->src);
}
}
tevent_req_received(req);
return ret;
}
struct tdgram_bsd_sendto_state {
struct tdgram_context *dgram;
const uint8_t *buf;
size_t len;
const struct tsocket_address *dst;
ssize_t ret;
};
static int tdgram_bsd_sendto_destructor(struct tdgram_bsd_sendto_state *state)
{
struct tdgram_bsd *bsds = tdgram_context_data(state->dgram,
struct tdgram_bsd);
bsds->write_req = NULL;
tdgram_bsd_set_writeable_handler(bsds, NULL, NULL, NULL);
return 0;
}
static void tdgram_bsd_sendto_handler(void *private_data);
static struct tevent_req *tdgram_bsd_sendto_send(TALLOC_CTX *mem_ctx,
struct tevent_context *ev,
struct tdgram_context *dgram,
const uint8_t *buf,
size_t len,
const struct tsocket_address *dst)
{
struct tevent_req *req;
struct tdgram_bsd_sendto_state *state;
struct tdgram_bsd *bsds = tdgram_context_data(dgram, struct tdgram_bsd);
int ret;
req = tevent_req_create(mem_ctx, &state,
struct tdgram_bsd_sendto_state);
if (!req) {
return NULL;
}
state->dgram = dgram;
state->buf = buf;
state->len = len;
state->dst = dst;
state->ret = -1;
if (bsds->write_req) {
tevent_req_error(req, EBUSY);
goto post;
}
bsds->write_req = req;
talloc_set_destructor(state, tdgram_bsd_sendto_destructor);
if (bsds->fd == -1) {
tevent_req_error(req, ENOTCONN);
goto post;
}
/*
* this is a fast path, not waiting for the
* socket to become explicit writeable gains
* about 10%-20% performance in benchmark tests.
*/
tdgram_bsd_sendto_handler(req);
if (!tevent_req_is_in_progress(req)) {
goto post;
}
ret = tdgram_bsd_set_writeable_handler(bsds, ev,
tdgram_bsd_sendto_handler,
req);
if (ret == -1) {
tevent_req_error(req, errno);
goto post;
}
return req;
post:
tevent_req_post(req, ev);
return req;
}
static void tdgram_bsd_sendto_handler(void *private_data)
{
struct tevent_req *req = talloc_get_type_abort(private_data,
struct tevent_req);
struct tdgram_bsd_sendto_state *state = tevent_req_data(req,
struct tdgram_bsd_sendto_state);
struct tdgram_context *dgram = state->dgram;
struct tdgram_bsd *bsds = tdgram_context_data(dgram, struct tdgram_bsd);
struct sockaddr *sa = NULL;
socklen_t sa_len = 0;
ssize_t ret;
int err;
bool retry;
if (state->dst) {
struct tsocket_address_bsd *bsda =
talloc_get_type(state->dst->private_data,
struct tsocket_address_bsd);
sa = &bsda->u.sa;
sa_len = sizeof(bsda->u.ss);
}
ret = sendto(bsds->fd, state->buf, state->len, 0, sa, sa_len);
err = tsocket_error_from_errno(ret, errno, &retry);
if (retry) {
/* retry later */
return;
}
if (tevent_req_error(req, err)) {
return;
}
state->ret = ret;
tevent_req_done(req);
}
static ssize_t tdgram_bsd_sendto_recv(struct tevent_req *req, int *perrno)
{
struct tdgram_bsd_sendto_state *state = tevent_req_data(req,
struct tdgram_bsd_sendto_state);
ssize_t ret;
ret = tsocket_simple_int_recv(req, perrno);
if (ret == 0) {
ret = state->ret;
}
tevent_req_received(req);
return ret;
}
struct tdgram_bsd_disconnect_state {
uint8_t __dummy;
};
static struct tevent_req *tdgram_bsd_disconnect_send(TALLOC_CTX *mem_ctx,
struct tevent_context *ev,
struct tdgram_context *dgram)
{
struct tdgram_bsd *bsds = tdgram_context_data(dgram, struct tdgram_bsd);
struct tevent_req *req;
struct tdgram_bsd_disconnect_state *state;
int ret;
int err;
bool dummy;
req = tevent_req_create(mem_ctx, &state,
struct tdgram_bsd_disconnect_state);
if (req == NULL) {
return NULL;
}
if (bsds->read_req || bsds->write_req) {
tevent_req_error(req, EBUSY);
goto post;
}
if (bsds->fd == -1) {
tevent_req_error(req, ENOTCONN);
goto post;
}
ret = close(bsds->fd);
bsds->fd = -1;
err = tsocket_error_from_errno(ret, errno, &dummy);
if (tevent_req_error(req, err)) {
goto post;
}
tevent_req_done(req);
post:
tevent_req_post(req, ev);
return req;
}
static int tdgram_bsd_disconnect_recv(struct tevent_req *req,
int *perrno)
{
int ret;
ret = tsocket_simple_int_recv(req, perrno);
tevent_req_received(req);
return ret;
}
static const struct tdgram_context_ops tdgram_bsd_ops = {
.name = "bsd",
.recvfrom_send = tdgram_bsd_recvfrom_send,
.recvfrom_recv = tdgram_bsd_recvfrom_recv,
.sendto_send = tdgram_bsd_sendto_send,
.sendto_recv = tdgram_bsd_sendto_recv,
.disconnect_send = tdgram_bsd_disconnect_send,
.disconnect_recv = tdgram_bsd_disconnect_recv,
};
static int tdgram_bsd_destructor(struct tdgram_bsd *bsds)
{
TALLOC_FREE(bsds->fde);
if (bsds->fd != -1) {
close(bsds->fd);
bsds->fd = -1;
}
return 0;
}
static int tdgram_bsd_dgram_socket(const struct tsocket_address *local,
const struct tsocket_address *remote,
TALLOC_CTX *mem_ctx,
struct tdgram_context **_dgram,
const char *location)
{
struct tsocket_address_bsd *lbsda =
talloc_get_type_abort(local->private_data,
struct tsocket_address_bsd);
struct tsocket_address_bsd *rbsda = NULL;
struct tdgram_context *dgram;
struct tdgram_bsd *bsds;
int fd;
int ret;
bool do_bind = false;
bool do_reuseaddr = false;
if (remote) {
rbsda = talloc_get_type_abort(remote->private_data,
struct tsocket_address_bsd);
}
switch (lbsda->u.sa.sa_family) {
case AF_UNIX:
if (lbsda->u.un.sun_path[0] != 0) {
do_reuseaddr = true;
do_bind = true;
}
break;
case AF_INET:
if (lbsda->u.in.sin_port != 0) {
do_reuseaddr = true;
do_bind = true;
}
if (lbsda->u.in.sin_addr.s_addr == INADDR_ANY) {
do_bind = true;
}
break;
#ifdef HAVE_IPV6
case AF_INET6:
if (lbsda->u.in6.sin6_port != 0) {
do_reuseaddr = true;
do_bind = true;
}
if (memcmp(&in6addr_any,
&lbsda->u.in6.sin6_addr,
sizeof(in6addr_any)) != 0) {
do_bind = true;
}
break;
#endif
default:
errno = EINVAL;
return -1;
}
fd = socket(lbsda->u.sa.sa_family, SOCK_DGRAM, 0);
if (fd < 0) {
return fd;
}
fd = tsocket_bsd_common_prepare_fd(fd, true);
if (fd < 0) {
return fd;
}
dgram = tdgram_context_create(mem_ctx,
&tdgram_bsd_ops,
&bsds,
struct tdgram_bsd,
location);
if (!dgram) {
int saved_errno = errno;
close(fd);
errno = saved_errno;
return -1;
}
ZERO_STRUCTP(bsds);
bsds->fd = fd;
talloc_set_destructor(bsds, tdgram_bsd_destructor);
if (lbsda->broadcast) {
int val = 1;
ret = setsockopt(fd, SOL_SOCKET, SO_BROADCAST,
(const void *)&val, sizeof(val));
if (ret == -1) {
int saved_errno = errno;
talloc_free(dgram);
errno = saved_errno;
return ret;
}
}
if (do_reuseaddr) {
int val = 1;
ret = setsockopt(fd, SOL_SOCKET, SO_REUSEADDR,
(const void *)&val, sizeof(val));
if (ret == -1) {
int saved_errno = errno;
talloc_free(dgram);
errno = saved_errno;
return ret;
}
}
if (do_bind) {
ret = bind(fd, &lbsda->u.sa, sizeof(lbsda->u.ss));
if (ret == -1) {
int saved_errno = errno;
talloc_free(dgram);
errno = saved_errno;
return ret;
}
}
if (rbsda) {
ret = connect(fd, &rbsda->u.sa, sizeof(rbsda->u.ss));
if (ret == -1) {
int saved_errno = errno;
talloc_free(dgram);
errno = saved_errno;
return ret;
}
}
*_dgram = dgram;
return 0;
}
int _tdgram_inet_udp_socket(const struct tsocket_address *local,
const struct tsocket_address *remote,
TALLOC_CTX *mem_ctx,
struct tdgram_context **dgram,
const char *location)
{
struct tsocket_address_bsd *lbsda =
talloc_get_type_abort(local->private_data,
struct tsocket_address_bsd);
int ret;
switch (lbsda->u.sa.sa_family) {
case AF_INET:
break;
#ifdef HAVE_IPV6
case AF_INET6:
break;
#endif
default:
errno = EINVAL;
return -1;
}
ret = tdgram_bsd_dgram_socket(local, remote, mem_ctx, dgram, location);
return ret;
}
int _tdgram_unix_dgram_socket(const struct tsocket_address *local,
const struct tsocket_address *remote,
TALLOC_CTX *mem_ctx,
struct tdgram_context **dgram,
const char *location)
{
struct tsocket_address_bsd *lbsda =
talloc_get_type_abort(local->private_data,
struct tsocket_address_bsd);
int ret;
switch (lbsda->u.sa.sa_family) {
case AF_UNIX:
break;
default:
errno = EINVAL;
return -1;
}
ret = tdgram_bsd_dgram_socket(local, remote, mem_ctx, dgram, location);
return ret;
}

View File

@ -57,14 +57,6 @@ struct tsocket_context_ops {
int (*writev_data)(struct tsocket_context *sock,
const struct iovec *vector, size_t count);
ssize_t (*recvfrom_data)(struct tsocket_context *sock,
uint8_t *data, size_t len,
TALLOC_CTX *addr_ctx,
struct tsocket_address **remote_addr);
ssize_t (*sendto_data)(struct tsocket_context *sock,
const uint8_t *data, size_t len,
const struct tsocket_address *remote_addr);
/* info */
int (*get_status)(const struct tsocket_context *sock);
int (*get_local_address)(const struct tsocket_context *sock,
@ -149,6 +141,47 @@ struct tsocket_address *_tsocket_address_create(TALLOC_CTX *mem_ctx,
_tsocket_address_create(mem_ctx, ops, state, sizeof(type), \
#type, location)
struct tdgram_context_ops {
const char *name;
struct tevent_req *(*recvfrom_send)(TALLOC_CTX *mem_ctx,
struct tevent_context *ev,
struct tdgram_context *dgram);
ssize_t (*recvfrom_recv)(struct tevent_req *req,
int *perrno,
TALLOC_CTX *mem_ctx,
uint8_t **buf,
struct tsocket_address **src);
struct tevent_req *(*sendto_send)(TALLOC_CTX *mem_ctx,
struct tevent_context *ev,
struct tdgram_context *dgram,
const uint8_t *buf, size_t len,
const struct tsocket_address *dst);
ssize_t (*sendto_recv)(struct tevent_req *req,
int *perrno);
struct tevent_req *(*disconnect_send)(TALLOC_CTX *mem_ctx,
struct tevent_context *ev,
struct tdgram_context *dgram);
int (*disconnect_recv)(struct tevent_req *req,
int *perrno);
};
struct tdgram_context *_tdgram_context_create(TALLOC_CTX *mem_ctx,
const struct tdgram_context_ops *ops,
void *pstate,
size_t psize,
const char *type,
const char *location);
#define tdgram_context_create(mem_ctx, ops, state, type, location) \
_tdgram_context_create(mem_ctx, ops, state, sizeof(type), \
#type, location)
void *_tdgram_context_data(struct tdgram_context *dgram);
#define tdgram_context_data(_req, _type) \
talloc_get_type_abort(_tdgram_context_data(_req), _type)
int tsocket_error_from_errno(int ret, int sys_errno, bool *retry);
int tsocket_simple_int_recv(struct tevent_req *req, int *perrno);
int tsocket_common_prepare_fd(int fd, bool high_fd);

View File

@ -1,164 +0,0 @@
/*
Unix SMB/CIFS implementation.
Copyright (C) Stefan Metzmacher 2009
** NOTE! The following LGPL license applies to the tevent
** library. This does NOT imply that all of Samba is released
** under the LGPL
This library is free software; you can redistribute it and/or
modify it under the terms of the GNU Lesser General Public
License as published by the Free Software Foundation; either
version 3 of the License, or (at your option) any later version.
This library is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
Lesser General Public License for more details.
You should have received a copy of the GNU Lesser General Public
License along with this library; if not, see <http://www.gnu.org/licenses/>.
*/
#include "replace.h"
#include "system/network.h"
#include "tsocket.h"
#include "tsocket_internal.h"
struct tsocket_recvfrom_state {
/* this structs are owned by the caller */
struct {
struct tsocket_context *sock;
} caller;
uint8_t *buf;
size_t len;
struct tsocket_address *src;
};
static int tsocket_recvfrom_state_destructor(struct tsocket_recvfrom_state *state)
{
if (state->caller.sock) {
tsocket_set_readable_handler(state->caller.sock, NULL, NULL);
}
ZERO_STRUCT(state->caller);
return 0;
}
static void tsocket_recvfrom_handler(struct tsocket_context *sock,
void *private_data);
struct tevent_req *tsocket_recvfrom_send(struct tsocket_context *sock,
TALLOC_CTX *mem_ctx)
{
struct tevent_req *req;
struct tsocket_recvfrom_state *state;
int ret;
int err;
bool dummy;
req = tevent_req_create(mem_ctx, &state,
struct tsocket_recvfrom_state);
if (!req) {
return NULL;
}
state->caller.sock = sock;
state->buf = NULL;
state->len = 0;
state->src = NULL;
talloc_set_destructor(state, tsocket_recvfrom_state_destructor);
ret = tsocket_set_readable_handler(sock,
tsocket_recvfrom_handler,
req);
err = tsocket_error_from_errno(ret, errno, &dummy);
if (tevent_req_error(req, err)) {
goto post;
}
return req;
post:
return tevent_req_post(req, sock->event.ctx);
}
static void tsocket_recvfrom_handler(struct tsocket_context *sock,
void *private_data)
{
struct tevent_req *req = talloc_get_type(private_data,
struct tevent_req);
struct tsocket_recvfrom_state *state = tevent_req_data(req,
struct tsocket_recvfrom_state);
ssize_t ret;
int err;
bool retry;
ret = tsocket_pending(state->caller.sock);
if (ret == 0) {
/* retry later */
return;
}
err = tsocket_error_from_errno(ret, errno, &retry);
if (retry) {
/* retry later */
return;
}
if (tevent_req_error(req, err)) {
return;
}
state->buf = talloc_array(state, uint8_t, ret);
if (tevent_req_nomem(state->buf, req)) {
return;
}
state->len = ret;
ret = tsocket_recvfrom(state->caller.sock,
state->buf,
state->len,
state,
&state->src);
err = tsocket_error_from_errno(ret, errno, &retry);
if (retry) {
/* retry later */
return;
}
if (tevent_req_error(req, err)) {
return;
}
if (ret != state->len) {
tevent_req_error(req, EIO);
return;
}
tevent_req_done(req);
}
ssize_t tsocket_recvfrom_recv(struct tevent_req *req,
int *perrno,
TALLOC_CTX *mem_ctx,
uint8_t **buf,
struct tsocket_address **src)
{
struct tsocket_recvfrom_state *state = tevent_req_data(req,
struct tsocket_recvfrom_state);
ssize_t ret;
ret = tsocket_simple_int_recv(req, perrno);
if (ret == 0) {
*buf = talloc_move(mem_ctx, &state->buf);
ret = state->len;
if (src) {
*src = talloc_move(mem_ctx, &state->src);
}
}
tevent_req_received(req);
return ret;
}

View File

@ -1,271 +0,0 @@
/*
Unix SMB/CIFS implementation.
Copyright (C) Stefan Metzmacher 2009
** NOTE! The following LGPL license applies to the tevent
** library. This does NOT imply that all of Samba is released
** under the LGPL
This library is free software; you can redistribute it and/or
modify it under the terms of the GNU Lesser General Public
License as published by the Free Software Foundation; either
version 3 of the License, or (at your option) any later version.
This library is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
Lesser General Public License for more details.
You should have received a copy of the GNU Lesser General Public
License along with this library; if not, see <http://www.gnu.org/licenses/>.
*/
#include "replace.h"
#include "system/network.h"
#include "tsocket.h"
#include "tsocket_internal.h"
struct tsocket_sendto_state {
/* this structs are owned by the caller */
struct {
struct tsocket_context *sock;
const uint8_t *buf;
size_t len;
const struct tsocket_address *dst;
} caller;
ssize_t ret;
};
static int tsocket_sendto_state_destructor(struct tsocket_sendto_state *state)
{
if (state->caller.sock) {
tsocket_set_writeable_handler(state->caller.sock, NULL, NULL);
}
ZERO_STRUCT(state->caller);
return 0;
}
static void tsocket_sendto_handler(struct tsocket_context *sock,
void *private_data);
struct tevent_req *tsocket_sendto_send(struct tsocket_context *sock,
TALLOC_CTX *mem_ctx,
const uint8_t *buf,
size_t len,
const struct tsocket_address *dst)
{
struct tevent_req *req;
struct tsocket_sendto_state *state;
int ret;
int err;
bool dummy;
req = tevent_req_create(mem_ctx, &state,
struct tsocket_sendto_state);
if (!req) {
return NULL;
}
state->caller.sock = sock;
state->caller.buf = buf;
state->caller.len = len;
state->caller.dst = dst;
state->ret = -1;
/*
* this is a fast path, not waiting for the
* socket to become explicit writeable gains
* about 10%-20% performance in benchmark tests.
*/
tsocket_sendto_handler(sock, req);
if (!tevent_req_is_in_progress(req)) {
goto post;
}
talloc_set_destructor(state, tsocket_sendto_state_destructor);
ret = tsocket_set_writeable_handler(sock,
tsocket_sendto_handler,
req);
err = tsocket_error_from_errno(ret, errno, &dummy);
if (tevent_req_error(req, err)) {
goto post;
}
return req;
post:
return tevent_req_post(req, sock->event.ctx);
}
static void tsocket_sendto_handler(struct tsocket_context *sock,
void *private_data)
{
struct tevent_req *req = talloc_get_type(private_data,
struct tevent_req);
struct tsocket_sendto_state *state = tevent_req_data(req,
struct tsocket_sendto_state);
ssize_t ret;
int err;
bool retry;
ret = tsocket_sendto(state->caller.sock,
state->caller.buf,
state->caller.len,
state->caller.dst);
err = tsocket_error_from_errno(ret, errno, &retry);
if (retry) {
/* retry later */
return;
}
if (tevent_req_error(req, err)) {
return;
}
state->ret = ret;
tevent_req_done(req);
}
ssize_t tsocket_sendto_recv(struct tevent_req *req, int *perrno)
{
struct tsocket_sendto_state *state = tevent_req_data(req,
struct tsocket_sendto_state);
ssize_t ret;
ret = tsocket_simple_int_recv(req, perrno);
if (ret == 0) {
ret = state->ret;
}
tevent_req_received(req);
return ret;
}
struct tsocket_sendto_queue_state {
/* this structs are owned by the caller */
struct {
struct tsocket_context *sock;
const uint8_t *buf;
size_t len;
const struct tsocket_address *dst;
} caller;
ssize_t ret;
};
static void tsocket_sendto_queue_trigger(struct tevent_req *req,
void *private_data);
static void tsocket_sendto_queue_done(struct tevent_req *subreq);
/**
* @brief Queue a dgram blob for sending through the socket
* @param[in] mem_ctx The memory context for the result
* @param[in] sock The socket to send the message buffer
* @param[in] queue The existing dgram queue
* @param[in] buf The message buffer
* @param[in] len The message length
* @param[in] dst The destination socket address
* @retval The async request handle
*
* This function queues a blob for sending to destination through an existing
* dgram socket. The async callback is triggered when the whole blob is
* delivered to the underlying system socket.
*
* The caller needs to make sure that all non-scalar input parameters hang
* arround for the whole lifetime of the request.
*/
struct tevent_req *tsocket_sendto_queue_send(TALLOC_CTX *mem_ctx,
struct tsocket_context *sock,
struct tevent_queue *queue,
const uint8_t *buf,
size_t len,
struct tsocket_address *dst)
{
struct tevent_req *req;
struct tsocket_sendto_queue_state *state;
bool ok;
req = tevent_req_create(mem_ctx, &state,
struct tsocket_sendto_queue_state);
if (!req) {
return NULL;
}
state->caller.sock = sock;
state->caller.buf = buf;
state->caller.len = len;
state->caller.dst = dst;
state->ret = -1;
ok = tevent_queue_add(queue,
sock->event.ctx,
req,
tsocket_sendto_queue_trigger,
NULL);
if (!ok) {
tevent_req_nomem(NULL, req);
goto post;
}
return req;
post:
return tevent_req_post(req, sock->event.ctx);
}
static void tsocket_sendto_queue_trigger(struct tevent_req *req,
void *private_data)
{
struct tsocket_sendto_queue_state *state = tevent_req_data(req,
struct tsocket_sendto_queue_state);
struct tevent_req *subreq;
subreq = tsocket_sendto_send(state->caller.sock,
state,
state->caller.buf,
state->caller.len,
state->caller.dst);
if (tevent_req_nomem(subreq, req)) {
return;
}
tevent_req_set_callback(subreq, tsocket_sendto_queue_done ,req);
}
static void tsocket_sendto_queue_done(struct tevent_req *subreq)
{
struct tevent_req *req = tevent_req_callback_data(subreq,
struct tevent_req);
struct tsocket_sendto_queue_state *state = tevent_req_data(req,
struct tsocket_sendto_queue_state);
ssize_t ret;
int sys_errno;
ret = tsocket_sendto_recv(subreq, &sys_errno);
talloc_free(subreq);
if (ret == -1) {
tevent_req_error(req, sys_errno);
return;
}
state->ret = ret;
tevent_req_done(req);
}
ssize_t tsocket_sendto_queue_recv(struct tevent_req *req, int *perrno)
{
struct tsocket_sendto_queue_state *state = tevent_req_data(req,
struct tsocket_sendto_queue_state);
ssize_t ret;
ret = tsocket_simple_int_recv(req, perrno);
if (ret == 0) {
ret = state->ret;
}
tevent_req_received(req);
return ret;
}

105
lib/util/smb_threads.c Normal file
View File

@ -0,0 +1,105 @@
/*
Unix SMB/CIFS implementation.
SMB client library implementation (thread interface functions).
Copyright (C) Jeremy Allison, 2009.
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
/*
* This code is based in the ideas in openssl
* but somewhat simpler and expended to include
* thread local storage.
*/
#include "includes.h"
#define NUM_GLOBAL_LOCKS 1
/*********************************************************
Functions to vector the locking primitives used internally
by libsmbclient.
*********************************************************/
const struct smb_thread_functions *global_tfp;
/*********************************************************
Dynamic lock array.
*********************************************************/
void **global_lock_array;
/*********************************************************
Function to set the locking primitives used by libsmbclient.
*********************************************************/
int smb_thread_set_functions(const struct smb_thread_functions *tf)
{
int i;
global_tfp = tf;
/* Here we initialize any static locks we're using. */
global_lock_array = (void **)SMB_MALLOC_ARRAY(void *, NUM_GLOBAL_LOCKS);
if (global_lock_array == NULL) {
return ENOMEM;
}
for (i = 0; i < NUM_GLOBAL_LOCKS; i++) {
char *name = NULL;
if (asprintf(&name, "global_lock_%d", i) == -1) {
SAFE_FREE(global_lock_array);
return ENOMEM;
}
global_tfp->create_mutex(name,
&global_lock_array[i],
__location__);
SAFE_FREE(name);
}
return 0;
}
#if 0
/* Test. - pthread implementations. */
#include <pthread.h>
#ifdef malloc
#undef malloc
#endif
SMB_THREADS_DEF_PTHREAD_IMPLEMENTATION(tf);
/* Test function. */
int test_threads(void)
{
int ret;
void *plock = NULL;
smb_thread_set_functions(&tf);
if ((ret = SMB_THREAD_CREATE_MUTEX("test", plock)) != 0) {
printf("Create lock error: %d\n", ret);
}
if ((ret = SMB_THREAD_LOCK(plock, SMB_THREAD_LOCK)) != 0) {
printf("lock error: %d\n", ret);
}
if ((SMB_THREAD_LOCK(plock, SMB_THREAD_UNLOCK)) != 0) {
printf("unlock error: %d\n", ret);
}
SMB_THREAD_DESTROY_MUTEX(plock);
return 0;
}
#endif

120
lib/util/smb_threads.h Normal file
View File

@ -0,0 +1,120 @@
/*
Unix SMB/CIFS implementation.
SMB thread interface functions.
Copyright (C) Jeremy Allison, 2009.
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
#ifndef _smb_threads_h_
#define _smb_threads_h_
enum smb_thread_lock_type {
SMB_THREAD_LOCK = 1,
SMB_THREAD_UNLOCK
};
struct smb_thread_functions {
/* Mutex and tls functions. */
int (*create_mutex)(const char *lockname,
void **pplock,
const char *location);
void (*destroy_mutex)(void *plock,
const char *location);
int (*lock_mutex)(void *plock, enum smb_thread_lock_type lock_type,
const char *location);
/* Thread local storage. */
int (*create_tls)(const char *keyname,
void **ppkey,
const char *location);
void (*destroy_tls)(void *pkey,
const char *location);
int (*set_tls)(void *pkey, const void *pval, const char *location);
void *(*get_tls)(void *pkey, const char *location);
};
extern const struct smb_thread_functions *global_tfp;
/* Define the pthread version of the functions. */
#define SMB_THREADS_DEF_PTHREAD_IMPLEMENTATION(tf) \
\
static int smb_create_mutex_pthread(const char *lockname, void **pplock, const char *location) \
{ \
pthread_mutex_t *pmut = (pthread_mutex_t *)malloc(sizeof(pthread_mutex_t)); \
if (!pmut) { \
return ENOMEM; \
} \
pthread_mutex_init(pmut, NULL); \
*pplock = (void *)pmut; \
return 0; \
} \
\
static void smb_destroy_mutex_pthread(void *plock, const char *location) \
{ \
pthread_mutex_destroy((pthread_mutex_t *)plock); \
free(plock); \
} \
\
static int smb_lock_pthread(void *plock, enum smb_thread_lock_type lock_type, const char *location) \
{ \
if (lock_type == SMB_THREAD_UNLOCK) { \
return pthread_mutex_unlock((pthread_mutex_t *)plock); \
} else { \
return pthread_mutex_lock((pthread_mutex_t *)plock); \
} \
} \
\
static int smb_create_tls_pthread(const char *keyname, void **ppkey, const char *location) \
{ \
int ret; \
pthread_key_t *pkey = (pthread_key_t *)malloc(sizeof(pthread_key_t)); \
if (!pkey) { \
return ENOMEM; \
} \
ret = pthread_key_create(pkey, NULL); \
if (ret) { \
return ret; \
} \
*ppkey = (void *)pkey; \
return 0; \
} \
\
static void smb_destroy_tls_pthread(void *pkey, const char *location) \
{ \
pthread_key_delete(*(pthread_key_t *)pkey); \
free(pkey); \
} \
\
static int smb_set_tls_pthread(void *pkey, const void *pval, const char *location) \
{ \
return pthread_setspecific(*(pthread_key_t *)pkey, pval); \
} \
\
static void *smb_get_tls_pthread(void *pkey, const char *location) \
{ \
return pthread_getspecific(*(pthread_key_t *)pkey); \
} \
\
static const struct smb_thread_functions (tf) = { \
smb_create_mutex_pthread, \
smb_destroy_mutex_pthread, \
smb_lock_pthread, \
smb_create_tls_pthread, \
smb_destroy_tls_pthread, \
smb_set_tls_pthread, \
smb_get_tls_pthread }
#endif

Some files were not shown because too many files have changed in this diff Show More