aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorLukasz Marek <lukasz.m.luki@gmail.com>2013-06-03 00:42:00 +0200
committerLukasz Marek <lukasz.m.luki@gmail.com>2013-06-08 04:17:26 +0200
commit4d617715c93d9f06ab38057639be6b5e0485fb3f (patch)
tree7792a1e87be46278174cbd9f35c91ed59715df17
parentbc29acdc76fdbf70700cdc2f85fc2afb46e19e47 (diff)
downloadffmpeg-4d617715c93d9f06ab38057639be6b5e0485fb3f.tar.gz
ftp: abort function optimalization
It seems some ftp servers doesn't respect ABOR command, but closing both connection is slow. This commit keeps control connection open when possible. Signed-off-by: Lukasz Marek <lukasz.m.luki@gmail.com>
-rw-r--r--libavformat/ftp.c64
1 files changed, 48 insertions, 16 deletions
diff --git a/libavformat/ftp.c b/libavformat/ftp.c
index b2e842f265..3c32fcd852 100644
--- a/libavformat/ftp.c
+++ b/libavformat/ftp.c
@@ -172,9 +172,6 @@ static int ftp_status(FTPContext *s, char **line, const int response_codes[])
return result;
}
- /* first code received. Now get all lines in non blocking mode */
- s->conn_control_block_flag = 1;
-
av_log(s, AV_LOG_DEBUG, "%s\n", buf);
if (!pref_code_found) {
@@ -191,6 +188,8 @@ static int ftp_status(FTPContext *s, char **line, const int response_codes[])
for (i = 0; response_codes[i]; ++i) {
if (err == response_codes[i]) {
+ /* first code received. Now get all lines in non blocking mode */
+ s->conn_control_block_flag = 1;
pref_code_found = 1;
result = err;
if (line)
@@ -216,19 +215,29 @@ static int ftp_send_command(FTPContext *s, const char *command,
s->conn_control_block_flag = 0;
if ((err = ffurl_write(s->conn_control, command, strlen(command))) < 0)
return err;
+ if (!err)
+ return -1;
/* return status */
- return ftp_status(s, response, response_codes);
+ if (response_codes) {
+ return ftp_status(s, response, response_codes);
+ }
+ return 0;
}
-static void ftp_close_both_connections(FTPContext *s)
+static void ftp_close_data_connection(FTPContext *s)
{
- ffurl_closep(&s->conn_control);
ffurl_closep(&s->conn_data);
s->position = 0;
s->state = DISCONNECTED;
}
+static void ftp_close_both_connections(FTPContext *s)
+{
+ ffurl_closep(&s->conn_control);
+ ftp_close_data_connection(s);
+}
+
static int ftp_auth(FTPContext *s)
{
const char *user = NULL, *pass = NULL;
@@ -445,8 +454,7 @@ static int ftp_connect_control_connection(URLContext *h)
/* consume all messages from server */
if (ftp_status(s, NULL, connect_codes) != 220) {
av_log(h, AV_LOG_ERROR, "FTP server not ready for new users\n");
- err = AVERROR(EACCES);
- return err;
+ return AVERROR(EACCES);
}
if ((err = ftp_auth(s)) < 0) {
@@ -497,12 +505,40 @@ static int ftp_connect_data_connection(URLContext *h)
static int ftp_abort(URLContext *h)
{
+ const char *command = "ABOR\r\n";
int err;
- ftp_close_both_connections(h->priv_data);
- if ((err = ftp_connect_control_connection(h)) < 0) {
- av_log(h, AV_LOG_ERROR, "Reconnect failed.\n");
- return err;
+ const int abor_codes[] = {225, 226, 0};
+ FTPContext *s = h->priv_data;
+
+ /* According to RCF 959:
+ "ABOR command tells the server to abort the previous FTP
+ service command and any associated transfer of data."
+
+ There are FTP server implementations that don't response
+ to any commands during data transfer in passive mode (including ABOR).
+
+ This implementation closes data connection by force.
+ */
+
+ if (ftp_send_command(s, command, NULL, NULL) < 0) {
+ ftp_close_both_connections(s);
+ if ((err = ftp_connect_control_connection(h)) < 0) {
+ av_log(h, AV_LOG_ERROR, "Reconnect failed.\n");
+ return err;
+ }
+ } else {
+ ftp_close_data_connection(s);
}
+
+ if (ftp_status(s, NULL, abor_codes) < 225) {
+ /* wu-ftpd also closes control connection after data connection closing */
+ ffurl_closep(&s->conn_control);
+ if ((err = ftp_connect_control_connection(h)) < 0) {
+ av_log(h, AV_LOG_ERROR, "Reconnect failed.\n");
+ return err;
+ }
+ }
+
return 0;
}
@@ -585,12 +621,8 @@ static int64_t ftp_seek(URLContext *h, int64_t pos, int whence)
new_pos = FFMIN(s->filesize, new_pos);
if (new_pos != s->position) {
- /* XXX: Full abort is a save solution here.
- Some optimalizations are possible, but may lead to crazy states of FTP server.
- The worst scenario would be when FTP server closed both connection due to no transfer. */
if ((err = ftp_abort(h)) < 0)
return err;
-
s->position = new_pos;
}
return new_pos;