スポンサーリンク

このドキュメントの内容は、以下の通りです。

Apacheの性能を向上させるための FreeBSDのカーネルモジュール Accept Filter(accf_http.ko) の中身が気になっていたので、適当に accf_http.c を読みました。

accp_http.c は、 FreeBSD の /usr/src/sys/netinet/accf_http.c にあります。
http://www.jp.freebsd.org/cgi/cvsweb.cgi/src/sys/netinet/accf_http.c

概要


socketプログラミングでは、接続を待ち受け、リクエストを読み込むタイプのデーモンは、accept()システムコールで接続を待ます。accept()が返ったら、今度は、リクエストを読み込むためにread()システムコールでデータを受信します。データが分割されて送信される場合には、何度もread()システムコールを呼び出す必要があります。

httpサーバの場合、accept()システムコールの呼び出しが返ったら、HTTPリクエストヘッダを読み込まなければなりません。
すべてのリクエストヘッダのデータ受信を終えるまでにread()システムコールを何度も呼び出してしまう場合、プロセス(httpd)とOSのスイッチングコンテキストが何度も発生してしまいます。このスイッチングは無駄以外の何者でもありません。

そこで、Accept Filterでは、HTTPリクエストヘッダが到着するまで、accept()が返らず、accept()システムコールが返ったときには、HTTPリクエストヘッダが到着しているので、1回のread()システムコールの呼び出しで、すべてのデータを受け取ることが可能になります。

流れ


sohashttpget()でリクエストの到着を待ちます。
GETかHEADのリクエストのみに対応しています。
parse_http_version(HTTPバージョンのパース処理)が1のときは、soparsehttpvers()を呼び出して、HTTPのバージョンを受信します。
その後、soishttpconnected()が呼び出され、すべてのリクエストヘッダの到着を待ちます。

今のところ HTTPパースバージョンは、HTTP/1.0かHTTP/1.1だけしか対応してません。
ハードコードされているので、バージョンが増えるたびに、ソースとバイナリを更新しないといけません。
もちろん、プロトコルバージョンが上がったときに、リクエストの内容に変更がおきれば、受信処理も変わってくるかもしれません。

sohashttpget()


"GET "
"HEAD "
のデータが到着したか確認する。
parse_http_versionが1なら
soishttpconnected()を呼び出す。
そうでなければ、
soparsehttpvers()を呼び出す。

soparsehttpvers()


"HTTP/1.0"
"HTTP/1.1"
のどちらかを受信していたらsoishttpconnected()を呼び出す。
"HTTP/1."までしかデータが届いてなかったら、再度呼び出されるようにして関数を終了する。

soishttpconnected()


\n\n か \n\r\n が到着するのを待つ。つまり、リクエストヘッダの終了を待つ。
到着してなければ、readmore処理に入る。

soisconnected()


soisconnected()は uipc_socket.c に定義されている。
/usr/src/sys/kern/uipc_socket.c
void soisconnected(struct socket *so);

メモ


GET/HEADのみの対応なので、POSTでは使えないということなのでしょうか。

とあるどっかの組織で聞いた話だとAccept FilterをLinuxにも移植しようとしたけど、mbufがないから、綺麗な実装ができなくてあきらめたとか言っていたような。

スポンサーリンク
スポンサーリンク
 
いつもシェア、ありがとうございます!


もっと情報を探しませんか?

関連記事

最近の記事

人気のページ

スポンサーリンク
 

過去ログ

2020 : 01 02 03 04 05 06 07 08 09 10 11 12
2019 : 01 02 03 04 05 06 07 08 09 10 11 12
2018 : 01 02 03 04 05 06 07 08 09 10 11 12
2017 : 01 02 03 04 05 06 07 08 09 10 11 12
2016 : 01 02 03 04 05 06 07 08 09 10 11 12
2015 : 01 02 03 04 05 06 07 08 09 10 11 12
2014 : 01 02 03 04 05 06 07 08 09 10 11 12
2013 : 01 02 03 04 05 06 07 08 09 10 11 12
2012 : 01 02 03 04 05 06 07 08 09 10 11 12
2011 : 01 02 03 04 05 06 07 08 09 10 11 12
2010 : 01 02 03 04 05 06 07 08 09 10 11 12
2009 : 01 02 03 04 05 06 07 08 09 10 11 12
2008 : 01 02 03 04 05 06 07 08 09 10 11 12
2007 : 01 02 03 04 05 06 07 08 09 10 11 12
2006 : 01 02 03 04 05 06 07 08 09 10 11 12
2005 : 01 02 03 04 05 06 07 08 09 10 11 12
2004 : 01 02 03 04 05 06 07 08 09 10 11 12
2003 : 01 02 03 04 05 06 07 08 09 10 11 12

サイト

Vim入門

C言語入門

C++入門

JavaScript/Node.js入門

Python入門

FreeBSD入門

Ubuntu入門

セキュリティ入門

パソコン自作入門

ブログ

トップ


プライバシーポリシー