Home | History | Annotate | Download | only in posix
      1 // Copyright (c) 2011 The Chromium Authors. All rights reserved.
      2 // Use of this source code is governed by a BSD-style license that can be
      3 // found in the LICENSE file.
      4 
      5 #include "base/posix/file_descriptor_shuffle.h"
      6 
      7 #include <unistd.h>
      8 #include <stddef.h>
      9 #include <ostream>
     10 
     11 #include "base/posix/eintr_wrapper.h"
     12 #include "base/logging.h"
     13 
     14 namespace base {
     15 
     16 bool PerformInjectiveMultimapDestructive(
     17     InjectiveMultimap* m, InjectionDelegate* delegate) {
     18   static const size_t kMaxExtraFDs = 16;
     19   int extra_fds[kMaxExtraFDs];
     20   unsigned next_extra_fd = 0;
     21 
     22   // DANGER: this function must not allocate or lock.
     23   // Cannot use STL iterators here, since debug iterators use locks.
     24 
     25   for (size_t i_index = 0; i_index < m->size(); ++i_index) {
     26     InjectiveMultimap::value_type* i = &(*m)[i_index];
     27     int temp_fd = -1;
     28 
     29     // We DCHECK the injectiveness of the mapping.
     30     for (size_t j_index = i_index + 1; j_index < m->size(); ++j_index) {
     31       InjectiveMultimap::value_type* j = &(*m)[j_index];
     32       DCHECK(i->dest != j->dest) << "Both fd " << i->source
     33           << " and " << j->source << " map to " << i->dest;
     34     }
     35 
     36     const bool is_identity = i->source == i->dest;
     37 
     38     for (size_t j_index = i_index + 1; j_index < m->size(); ++j_index) {
     39       InjectiveMultimap::value_type* j = &(*m)[j_index];
     40       if (!is_identity && i->dest == j->source) {
     41         if (temp_fd == -1) {
     42           if (!delegate->Duplicate(&temp_fd, i->dest))
     43             return false;
     44           if (next_extra_fd < kMaxExtraFDs) {
     45             extra_fds[next_extra_fd++] = temp_fd;
     46           } else {
     47             RAW_LOG(ERROR, "PerformInjectiveMultimapDestructive overflowed "
     48                            "extra_fds. Leaking file descriptors!");
     49           }
     50         }
     51 
     52         j->source = temp_fd;
     53         j->close = false;
     54       }
     55 
     56       if (i->close && i->source == j->dest)
     57         i->close = false;
     58 
     59       if (i->close && i->source == j->source) {
     60         i->close = false;
     61         j->close = true;
     62       }
     63     }
     64 
     65     if (!is_identity) {
     66       if (!delegate->Move(i->source, i->dest))
     67         return false;
     68     }
     69 
     70     if (!is_identity && i->close)
     71       delegate->Close(i->source);
     72   }
     73 
     74   for (unsigned i = 0; i < next_extra_fd; i++)
     75     delegate->Close(extra_fds[i]);
     76 
     77   return true;
     78 }
     79 
     80 bool PerformInjectiveMultimap(const InjectiveMultimap& m_in,
     81                               InjectionDelegate* delegate) {
     82   InjectiveMultimap m(m_in);
     83   return PerformInjectiveMultimapDestructive(&m, delegate);
     84 }
     85 
     86 bool FileDescriptorTableInjection::Duplicate(int* result, int fd) {
     87   *result = HANDLE_EINTR(dup(fd));
     88   return *result >= 0;
     89 }
     90 
     91 bool FileDescriptorTableInjection::Move(int src, int dest) {
     92   return HANDLE_EINTR(dup2(src, dest)) != -1;
     93 }
     94 
     95 void FileDescriptorTableInjection::Close(int fd) {
     96   int ret = IGNORE_EINTR(close(fd));
     97   DPCHECK(ret == 0);
     98 }
     99 
    100 }  // namespace base
    101