Commit c3bd8d14 authored by Nick Mathewson's avatar Nick Mathewson 🦀
Browse files

r18031@catbus: nickm | 2008-02-11 13:54:58 -0500

 Have assert_connection_ok() allow marked-for-close dir conns with stuff to flush but no way to flush it.  Adjust conn_dirserv_unlink_from_bridge() to mark edge and dir conns not already marked, since once linked conns are unlinked they are no longer viable.  Likely fix for bug 406, which was crashing 0.1.2.x servers periodically.


svn:r13469
parent 259d2f72
Loading
Loading
Loading
Loading
+4 −0
Original line number Diff line number Diff line
@@ -3,6 +3,10 @@ Changes in versino 0.1.2.20 - 2008-??-??
    - Patch from "Andrew S. Lists" to catch when we contact a directory
      mirror at IP address X and he says we look like we're coming from
      IP address X. Bugfix on 0.1.2.x.
    - Allow a closing-down linked directory connection to have its
      blocked_on_or_conn field set.  This prevents a rare assertion error
      that could occur when an OR connection carrying tunneled directory
      requests closed before the requests were complete.  Fixes bug 406.

  o Minor bugfixes:
    - Stop recommending that every server operator send mail to tor-ops.
+2 −1
Original line number Diff line number Diff line
@@ -2403,7 +2403,8 @@ assert_connection_ok(connection_t *conn, time_t now)
  if (conn->outbuf_flushlen > 0) {
    tor_assert(connection_is_writing(conn) || conn->wants_to_write ||
               (conn->type == CONN_TYPE_DIR &&
                TO_DIR_CONN(conn)->is_blocked_on_or_conn));
                (conn->marked_for_close ||
                 TO_DIR_CONN(conn)->is_blocked_on_or_conn)));
  }

  if (conn->hold_open_until_flushed)
+8 −2
Original line number Diff line number Diff line
@@ -2051,12 +2051,18 @@ connection_dirserv_unlink_from_bridge(dir_connection_t *dir_conn)
  or_conn = connection_dirserv_get_target_or_conn(dir_conn);
  if (or_conn) {
    /* XXXX Really, this is only necessary if dir_conn->is_blocked_on_or_conn.
     * But for now, let's leave it in, so the assert can catch  */
     * But for now, let's leave it in, so the assert can catch problems.  */
    connection_dirserv_remove_from_blocked_list(or_conn, dir_conn);
  }
  dir_conn->is_blocked_on_or_conn = 0; /* Probably redundant. */
  edge_conn->bridge_for_conn = NULL;
  dir_conn->bridge_conn = NULL;
  if (edge_conn) {
    edge_conn->bridge_for_conn = NULL;
    if (!edge_conn->_base.marked_for_close)
      connection_mark_for_close(TO_CONN(edge_conn));
  }
  if (!dir_conn->_base.marked_for_close)
    connection_mark_for_close(TO_CONN(dir_conn));
}

/** Stop writing on a bridged dir_conn, and remember that it's blocked because