admin
2022-10-21 0bda9070e5f7dc78ab7e6e394818f6dc82967623
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
// Copyright 2017 The Crashpad Authors. All rights reserved.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
//     http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
 
#ifndef CRASHPAD_UTIL_LINUX_MEMORY_MAP_H_
#define CRASHPAD_UTIL_LINUX_MEMORY_MAP_H_
 
#include <sys/types.h>
 
#include <memory>
#include <string>
#include <vector>
 
#include "util/linux/address_types.h"
#include "util/linux/checked_linux_address_range.h"
#include "util/linux/ptrace_connection.h"
#include "util/misc/initialization_state_dcheck.h"
 
namespace crashpad {
 
//! \brief Accesses information about mapped memory in another process.
//!
//! The target process must be stopped to guarantee correct mappings. If the
//! target process is not stopped, mappings may be invalid after the return from
//! Initialize(), and even mappings existing at the time Initialize() was called
//! may not be found.
class MemoryMap {
 public:
  //! \brief Information about a mapped region of memory.
  struct Mapping {
    Mapping();
    bool Equals(const Mapping& other) const;
 
    std::string name;
    CheckedLinuxAddressRange range;
    off64_t offset;
    dev_t device;
    ino_t inode;
    bool readable;
    bool writable;
    bool executable;
    bool shareable;
  };
 
  MemoryMap();
  ~MemoryMap();
 
  //! \brief Initializes this object with information about the mapped memory
  //!     regions in the process connected via \a connection.
  //!
  //! This method must be called successfully prior to calling any other method
  //! in this class. This method may only be called once.
  //!
  //! \param[in] connection A connection to the process create a map for.
  //!
  //! \return `true` on success, `false` on failure with a message logged.
  bool Initialize(PtraceConnection* connection);
 
  //! \return The Mapping containing \a address or `nullptr` if no match is
  //!     found. The caller does not take ownership of this object. It is scoped
  //!     to the lifetime of the MemoryMap object that it was obtained from.
  const Mapping* FindMapping(LinuxVMAddress address) const;
 
  //! \return The Mapping with the lowest base address whose name is \a name or
  //!     `nullptr` if no match is found. The caller does not take ownership of
  //!     this object. It is scoped to the lifetime of the MemoryMap object that
  //!     it was obtained from.
  const Mapping* FindMappingWithName(const std::string& name) const;
 
  //! \brief An abstract base class for iterating over ordered sets of mappings
  //!   in a MemoryMap.
  class Iterator {
   public:
    virtual ~Iterator() = default;
 
    //! \return the mapping pointed to by the iterator and advance the iterator
    //!     to the next mapping. If there are no more mappings, this method
    //!     returns `nullptr` on all subsequent invocations.
    virtual const Mapping* Next() = 0;
 
    //! \return the number of mappings remaining.
    virtual unsigned int Count() = 0;
 
   protected:
    Iterator() = default;
  };
 
  //! \brief Find possible initial mappings of files mapped over several
  //!     segments.
  //!
  //! Executables and libaries are typically loaded into several mappings with
  //! varying permissions for different segments. Portions of an ELF file may
  //! be mapped multiple times as part of loading the file, for example, when
  //! initializing GNU_RELRO segments.
  //!
  //! This method searches for mappings at or below \a mapping in memory that
  //! are mapped from the same file as \a mapping from offset 0.
  //!
  //! On Android, ELF modules may be loaded from within a zipfile, so this
  //! method may return mappings whose offset is not 0.
  //!
  //! This method is intended to help identify the possible base address for
  //! loaded modules, but it is the caller's responsibility to determine which
  //! returned mapping is correct.
  //!
  //! If \a mapping does not refer to a valid mapping, an empty vector will be
  //! returned and a message will be logged. If \a mapping is found but does not
  //! map a file, \a mapping is returned in \a possible_starts.
  //!
  //! \param[in] mapping A Mapping whose series to find the start of.
  //! \return a reverse iterator over the possible mapping starts, starting from
  //!     the mapping with highest base address.
  std::unique_ptr<Iterator> FindFilePossibleMmapStarts(
      const Mapping& mapping) const;
 
  //! \return A reverse iterator over all mappings in the MemoryMap from \a
  //!     mapping to the start of the MemoryMap.
  std::unique_ptr<Iterator> ReverseIteratorFrom(const Mapping& mapping) const;
 
 private:
  std::vector<Mapping> mappings_;
  InitializationStateDcheck initialized_;
};
 
}  // namespace crashpad
 
#endif  // CRASHPAD_UTIL_LINUX_MEMORY_MAP_H_