99f6a0
--- a/src/SFtp.cc	
99f6a0
+++ a/src/SFtp.cc	
99f6a0
@@ -299,12 +299,7 @@ void SFtp::MoveConnectionHere(SFtp *o)
99f6a0
    recv_translate=o->recv_translate.borrow();
99f6a0
    send_translate=o->send_translate.borrow();
99f6a0
    rate_limit=o->rate_limit.borrow();
99f6a0
-   expect_queue_size=o->expect_queue_size; o->expect_queue_size=0;
99f6a0
-   expect_chain=o->expect_chain; o->expect_chain=0;
99f6a0
-   expect_chain_end=o->expect_chain_end;
99f6a0
-   if(expect_chain_end==&o->expect_chain)
99f6a0
-      expect_chain_end=&expect_chain;
99f6a0
-   o->expect_chain_end=&o->expect_chain;
99f6a0
+   expect_queue.move_here(o->expect_queue);
99f6a0
    timeout_timer.Reset(o->timeout_timer);
99f6a0
    ssh_id=o->ssh_id;
99f6a0
    state=CONNECTED;
99f6a0
@@ -337,10 +332,6 @@ void SFtp::Init()
99f6a0
    eof=false;
99f6a0
    received_greeting=false;
99f6a0
    password_sent=0;
99f6a0
-   expect_queue_size=0;
99f6a0
-   expect_chain=0;
99f6a0
-   expect_chain_end=&expect_chain;
99f6a0
-   ooo_chain=0;
99f6a0
    protocol_version=0;
99f6a0
    send_translate=0;
99f6a0
    recv_translate=0;
99f6a0
@@ -693,8 +684,7 @@ void SFtp::Close()
99f6a0
    CloseHandle(Expect::IGNORE);
99f6a0
    super::Close();
99f6a0
    // don't need these out-of-order packets anymore
99f6a0
-   while(ooo_chain)
99f6a0
-      DeleteExpect(&ooo_chain);
99f6a0
+   ooo_chain.truncate();
99f6a0
    if(recv_buf)
99f6a0
       recv_buf->Resume();
99f6a0
 }
99f6a0
@@ -796,7 +786,7 @@ void SFtp::HandleExpect(Expect *e)
99f6a0
 	    SetError(NO_FILE,strerror(ENOTDIR));
99f6a0
 	    break;
99f6a0
 	 }
99f6a0
-	 if(mode==CHANGE_DIR && RespQueueIsEmpty())
99f6a0
+	 if(mode==CHANGE_DIR && GetExpectCount(Expect::CWD)==0)
99f6a0
 	 {
99f6a0
 	    cwd.Set(file);
99f6a0
 	    eof=true;
99f6a0
@@ -869,11 +859,15 @@ void SFtp::HandleExpect(Expect *e)
99f6a0
 	 }
99f6a0
 	 else
99f6a0
 	 {
99f6a0
-	    if(e->next!=ooo_chain)
99f6a0
-	       LogNote(9,"put a packet with id=%d on out-of-order chain (need_pos=%lld packet_pos=%lld)",
99f6a0
-		  reply->GetID(),(long long)(pos+file_buf->Size()),(long long)r->pos);
99f6a0
-	    e->next=ooo_chain;
99f6a0
-	    ooo_chain=e;
99f6a0
+	    LogNote(9,"put a packet with id=%d on out-of-order chain (need_pos=%lld packet_pos=%lld)",
99f6a0
+	       reply->GetID(),(long long)(pos+file_buf->Size()),(long long)r->pos);
99f6a0
+	    if(ooo_chain.count()>=64)
99f6a0
+	    {
99f6a0
+	       LogError(0,"Too many out-of-order packets");
99f6a0
+	       Disconnect();
99f6a0
+	       return;
99f6a0
+	    }
99f6a0
+	    ooo_chain.append(e);
99f6a0
 	    return;
99f6a0
 	 }
99f6a0
       }
99f6a0
@@ -925,7 +919,7 @@ void SFtp::HandleExpect(Expect *e)
99f6a0
 		  LogNote(9,"eof");
99f6a0
 	       eof=true;
99f6a0
 	       state=DONE;
99f6a0
-	       if(file_buf && !ooo_chain)
99f6a0
+	       if(file_buf && ooo_chain.count()==0)
99f6a0
 		  file_buf->PutEOF();
99f6a0
 	       break;
99f6a0
 	    }
99f6a0
@@ -1006,22 +1000,20 @@ int SFtp::HandleReplies()
99f6a0
    if(!recv_buf)
99f6a0
       return MOVED;
99f6a0
 
99f6a0
-   int i=0;
99f6a0
-   Expect *ooo_scan=ooo_chain;
99f6a0
-   while(ooo_scan)
99f6a0
-   {
99f6a0
-      Expect *next=ooo_scan->next;
99f6a0
-      ooo_chain=next;
99f6a0
-      HandleExpect(ooo_scan);
99f6a0
-      ooo_scan=next;
99f6a0
-      if(++i>64)
99f6a0
-      {
99f6a0
-	 LogError(0,"Too many out-of-order packets");
99f6a0
-	 Disconnect();
99f6a0
-	 return MOVED;
99f6a0
+   if(file_buf) {
99f6a0
+      off_t need_pos=pos+file_buf->Size();
99f6a0
+      // there are usually a few of out-of-order packets, no need for fast search
99f6a0
+      for(int i=0; i
99f6a0
+	 if(ooo_chain[i]->has_data_at_pos(need_pos)) {
99f6a0
+	    Expect *e=ooo_chain[i];
99f6a0
+	    ooo_chain[i]=0; // to keep the Expect
99f6a0
+	    ooo_chain.remove(i);
99f6a0
+	    HandleExpect(e);
99f6a0
+	 }
99f6a0
       }
99f6a0
    }
99f6a0
-   if(!ooo_chain && eof && file_buf && !file_buf->Eof())
99f6a0
+
99f6a0
+   if(ooo_chain.count()==0 && eof && file_buf && !file_buf->Eof())
99f6a0
       file_buf->PutEOF();
99f6a0
 
99f6a0
    if(recv_buf->Size()<4)
99f6a0
@@ -1066,51 +1058,20 @@ int SFtp::HandleReplies()
99f6a0
    HandleExpect(e);
99f6a0
    return MOVED;
99f6a0
 }
99f6a0
-SFtp::Expect **SFtp::FindExpect(Packet *p)
99f6a0
-{
99f6a0
-   unsigned id=p->GetID();
99f6a0
-   for(Expect **scan=&expect_chain; *scan; scan=&scan[0]->next)
99f6a0
-   {
99f6a0
-      if(scan[0]->request->GetID()==id)
99f6a0
-      {
99f6a0
-	 assert(!scan[0]->reply);
99f6a0
-	 scan[0]->reply=p;
99f6a0
-	 return scan;
99f6a0
-      }
99f6a0
-   }
99f6a0
-   return 0;
99f6a0
-}
99f6a0
 void SFtp::PushExpect(Expect *e)
99f6a0
 {
99f6a0
-   e->next=*expect_chain_end;
99f6a0
-   *expect_chain_end=e;
99f6a0
-   expect_chain_end=&e->next;
99f6a0
-   expect_queue_size++;
99f6a0
-}
99f6a0
-void SFtp::DeleteExpect(Expect **e)
99f6a0
-{
99f6a0
-   if(expect_chain_end==&e[0]->next)
99f6a0
-      expect_chain_end=e;
99f6a0
-   Expect *d=*e;
99f6a0
-   *e=e[0]->next;
99f6a0
-   delete d;
99f6a0
-   expect_queue_size--;
99f6a0
+   expect_queue.add(e->request->GetKey(),e);
99f6a0
 }
99f6a0
 SFtp::Expect *SFtp::FindExpectExclusive(Packet *p)
99f6a0
 {
99f6a0
-   Expect **e=FindExpect(p);
99f6a0
-   if(!e || !*e)
99f6a0
-      return 0;
99f6a0
-   Expect *res=*e;
99f6a0
-   if(expect_chain_end==&res->next)
99f6a0
-      expect_chain_end=e;
99f6a0
-   *e=res->next;
99f6a0
-   expect_queue_size--;
99f6a0
-   return res;
99f6a0
+   Expect *e=expect_queue.borrow(p->GetKey());
99f6a0
+   if(e)
99f6a0
+      e->reply=p;
99f6a0
+   return e;
99f6a0
 }
99f6a0
 void SFtp::CloseExpectQueue()
99f6a0
 {
99f6a0
-   for(Expect  *e=expect_chain; e; e=e->next)
99f6a0
+   for(Expect *e=expect_queue.each_begin(); e; e=expect_queue.each_next())
99f6a0
    {
99f6a0
       switch(e->tag)
99f6a0
       {
99f6a0
@@ -1133,6 +1094,14 @@ void SFtp::CloseExpectQueue()
99f6a0
    }
99f6a0
 }
99f6a0
 
99f6a0
+int SFtp::GetExpectCount(Expect::expect_t tag)
99f6a0
+{
99f6a0
+   int count=0;
99f6a0
+   for(Expect *e=expect_queue.each_begin(); e; e=expect_queue.each_next())
99f6a0
+      count+=(e->tag==tag);
99f6a0
+   return count;
99f6a0
+}
99f6a0
+
99f6a0
 Glob *SFtp::MakeGlob(const char *pat)
99f6a0
 {
99f6a0
    return new GenericGlob(this,pat);
99f6a0
--- a/src/SFtp.h	
99f6a0
+++ a/src/SFtp.h	
99f6a0
@@ -28,6 +28,7 @@ 
99f6a0
 #include <sys/types.h>
99f6a0
 #include <sys/stat.h>
99f6a0
 #include "FileSet.h"
99f6a0
+#include "xmap.h"
99f6a0
 
99f6a0
 class SFtp : public SSH_Access
99f6a0
 {
99f6a0
@@ -291,6 +292,7 @@ public:
99f6a0
       packet_type GetPacketType() { return type; }
99f6a0
       const char *GetPacketTypeText();
99f6a0
       unsigned GetID() const { return id; }
99f6a0
+      const xstring& GetKey() { return xstring::get_tmp((const char*)&id,sizeof(id)); }
99f6a0
       void SetID(unsigned new_id) { id=new_id; }
99f6a0
       void DropData(Buffer *b) { b->Skip(4+(length>0?length:0)); }
99f6a0
       bool TypeIs(packet_type t) const { return type==t; }
99f6a0
@@ -674,10 +676,15 @@ private:
99f6a0
 
99f6a0
       Ref<Packet> request;
99f6a0
       Ref<Packet> reply;
99f6a0
-      Expect *next;
99f6a0
       int i;
99f6a0
       expect_t tag;
99f6a0
       Expect(Packet *req,expect_t t,int j=0) : request(req), i(j), tag(t) {}
99f6a0
+
99f6a0
+      bool has_data_at_pos(off_t pos) const {
99f6a0
+	 if(!reply->TypeIs(SSH_FXP_DATA) || !request->TypeIs(SSH_FXP_READ))
99f6a0
+	    return false;
99f6a0
+	 return request.Cast<Request_READ>()->pos==pos;
99f6a0
+      }
99f6a0
    };
99f6a0
 
99f6a0
    void PushExpect(Expect *);
99f6a0
@@ -685,26 +692,22 @@ private:
99f6a0
    int HandlePty();
99f6a0
    void HandleExpect(Expect *);
99f6a0
    void CloseExpectQueue();
99f6a0
+   int GetExpectCount(Expect::expect_t tag);
99f6a0
    void CloseHandle(Expect::expect_t e);
99f6a0
    int ReplyLogPriority(int);
99f6a0
 
99f6a0
-   int expect_queue_size;
99f6a0
-   Expect *expect_chain;
99f6a0
-   Expect **expect_chain_end;
99f6a0
-   Expect **FindExpect(Packet *reply);
99f6a0
-   void DeleteExpect(Expect **);
99f6a0
-   Expect *FindExpectExclusive(Packet *reply);
99f6a0
-   Expect *ooo_chain; 	// out of order replies buffered
99f6a0
+   xmap_p<Expect> expect_queue;
99f6a0
+   const xstring& expect_key(unsigned id);
99f6a0
 
99f6a0
-   int   RespQueueIsEmpty() { return expect_chain==0; }
99f6a0
-   int	 RespQueueSize() { return expect_queue_size; }
99f6a0
-   void  EmptyRespQueue()
99f6a0
-      {
99f6a0
-	 while(expect_chain)
99f6a0
-	    DeleteExpect(&expect_chain);
99f6a0
-	 while(ooo_chain)
99f6a0
-	    DeleteExpect(&ooo_chain);
99f6a0
-      }
99f6a0
+   Expect *FindExpectExclusive(Packet *reply);
99f6a0
+   xarray_p<Expect> ooo_chain; 	// out of order replies buffered
99f6a0
+
99f6a0
+   int	 RespQueueSize() const { return expect_queue.count(); }
99f6a0
+   int   RespQueueIsEmpty() const { return RespQueueSize()==0; }
99f6a0
+   void  EmptyRespQueue() {
99f6a0
+      expect_queue.empty();
99f6a0
+      ooo_chain.truncate();
99f6a0
+   }
99f6a0
 
99f6a0
    bool GetBetterConnection(int level,bool limit_reached);
99f6a0
    void MoveConnectionHere(SFtp *o);
99f6a0
--- a/src/xmap.cc
99f6a0
+++ a/src/xmap.cc
99f6a0
@@ -133,8 +137,9 @@ void _xmap::_remove(entry **ep)
99f6a0
    if(!ep || !*ep)
99f6a0
       return;
99f6a0
    entry *e=*ep;
99f6a0
+   e->key.unset();
99f6a0
    *ep=e->next;
99f6a0
-   delete e;
99f6a0
+   xfree(e);
99f6a0
    entry_count--;
99f6a0
 }
99f6a0
 
99f6a0
--- a/src/xmap.h
99f6a0
+++ b/src/xmap.h
99f6a0
@@ -103,6 +103,7 @@ template<class T> T xmap<T>::zero;
99f6a0
 
99f6a0
 template<class T> class xmap_p : public _xmap
99f6a0
 {
99f6a0
+void dispose(T *p) { delete p; }
99f6a0
 public:
99f6a0
    xmap_p() : _xmap(sizeof(T*)) {}
99f6a0
    ~xmap_p() {
99f6a0
@@ -136,7 +137,7 @@ public:
99f6a0
    }
99f6a0
    void add(const xstring& key,T *e0) {
99f6a0
       entry *e=_add(key);
99f6a0
-      delete(payload(e));
99f6a0
+      dispose(payload(e));
99f6a0
       payload_Lv(e)=e0;
99f6a0
    }
99f6a0
    void add(const char *key,T *e0) { add(xstring::get_tmp(key),e0); }
99f6a0
@@ -148,7 +148,7 @@ public:
99f6a0
    void empty() {
99f6a0
       for(int i=0; i
99f6a0
 	 while(map[i]) {
99f6a0
-	    delete(payload(map[i]));
99f6a0
+	    dispose(payload(map[i]));
99f6a0
 	    _remove(&map[i]);
99f6a0
 	 }
99f6a0
       }