/* * Copyright (C) 2019 The Android Open Source Project * * 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. */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include namespace android { namespace dmabufinfo { static bool FileIsDmaBuf(const std::string& path) { return ::android::base::StartsWith(path, "/dmabuf"); } static bool ReadDmaBufFdInfo(pid_t pid, int fd, std::string* name, std::string* exporter, uint64_t* count) { std::string fdinfo = ::android::base::StringPrintf("/proc/%d/fdinfo/%d", pid, fd); auto fp = std::unique_ptr{fopen(fdinfo.c_str(), "re"), fclose}; if (fp == nullptr) { LOG(ERROR) << "Failed to open dmabuf info from debugfs"; return false; } char* line = nullptr; size_t len = 0; while (getline(&line, &len, fp.get()) > 0) { switch (line[0]) { case 'c': if (strncmp(line, "count:", 6) == 0) { char* c = line + 6; *count = strtoull(c, nullptr, 10); } break; case 'e': if (strncmp(line, "exp_name:", 9) == 0) { char* c = line + 9; *exporter = ::android::base::Trim(c); } break; case 'n': if (strncmp(line, "name:", 5) == 0) { char* c = line + 5; *name = ::android::base::Trim(std::string(c)); } break; } } free(line); return true; } // TODO: std::filesystem::is_symlink fails to link on vendor code, // forcing this workaround. // Move back to libc++fs once it is vendor-available. See b/124012728 static bool is_symlink(const char *filename) { struct stat p_statbuf; if (lstat(filename, &p_statbuf) < 0) { return false; } if (S_ISLNK(p_statbuf.st_mode) == 1) { return true; } return false; } static bool ReadDmaBufFdRefs(pid_t pid, std::vector* dmabufs) { std::string fdpath = ::android::base::StringPrintf("/proc/%d/fd", pid); std::unique_ptr dir(opendir(fdpath.c_str()), closedir); if (!dir) { LOG(ERROR) << "Failed to open " << fdpath << " directory" << std::endl; return false; } struct dirent* dent; while ((dent = readdir(dir.get()))) { std::string path = ::android::base::StringPrintf("%s/%s", fdpath.c_str(), dent->d_name); if (!strcmp(dent->d_name, ".") || !strcmp(dent->d_name, "..") || !is_symlink(path.c_str())) { continue; } std::string target; if (!::android::base::Readlink(path, &target)) { LOG(ERROR) << "Failed to find target for symlink: " << path; return false; } if (!FileIsDmaBuf(target)) { continue; } int fd; if (!::android::base::ParseInt(dent->d_name, &fd)) { LOG(ERROR) << "Dmabuf fd: " << path << " is invalid"; return false; } // Set defaults in case the kernel doesn't give us the information // we need in fdinfo std::string name = ""; std::string exporter = ""; uint64_t count = 0; if (!ReadDmaBufFdInfo(pid, fd, &name, &exporter, &count)) { LOG(ERROR) << "Failed to read fdinfo for: " << path; return false; } struct stat sb; if (stat(path.c_str(), &sb) < 0) { PLOG(ERROR) << "Failed to stat: " << path; return false; } uint64_t inode = sb.st_ino; auto buf = std::find_if(dmabufs->begin(), dmabufs->end(), [&inode](const DmaBuffer& dbuf) { return dbuf.inode() == inode; }); if (buf != dmabufs->end()) { if (buf->name() == "" || buf->name() == "") buf->SetName(name); if (buf->exporter() == "" || buf->exporter() == "") buf->SetExporter(exporter); if (buf->count() == 0) buf->SetCount(count); buf->AddFdRef(pid); continue; } DmaBuffer& db = dmabufs->emplace_back(sb.st_ino, sb.st_blocks * 512, count, exporter, name); db.AddFdRef(pid); } return true; } static bool ReadDmaBufMapRefs(pid_t pid, std::vector* dmabufs) { std::string mapspath = ::android::base::StringPrintf("/proc/%d/maps", pid); auto fp = std::unique_ptr{fopen(mapspath.c_str(), "re"), fclose}; if (fp == nullptr) { LOG(ERROR) << "Failed to open maps for pid: " << pid; return false; } char* line = nullptr; size_t len = 0; // Process the map if it is dmabuf. Add map reference to existing object in 'dmabufs' // if it was already found. If it wasn't create a new one and append it to 'dmabufs' auto account_dmabuf = [&](uint64_t start, uint64_t end, uint16_t /* flags */, uint64_t /* pgoff */, ino_t inode, const char* name) { // no need to look into this mapping if it is not dmabuf if (!FileIsDmaBuf(std::string(name))) { return; } auto buf = std::find_if(dmabufs->begin(), dmabufs->end(), [&inode](const DmaBuffer& dbuf) { return dbuf.inode() == inode; }); if (buf != dmabufs->end()) { buf->AddMapRef(pid); return; } // We have a new buffer, but unknown count and name DmaBuffer& dbuf = dmabufs->emplace_back(inode, end - start, 0, "", ""); dbuf.AddMapRef(pid); }; while (getline(&line, &len, fp.get()) > 0) { if (!::android::procinfo::ReadMapFileContent(line, account_dmabuf)) { LOG(ERROR) << "Failed t parse maps for pid: " << pid; return false; } } free(line); return true; } // Public methods bool ReadDmaBufInfo(std::vector* dmabufs, const std::string& path) { auto fp = std::unique_ptr{fopen(path.c_str(), "re"), fclose}; if (fp == nullptr) { LOG(ERROR) << "Failed to open dmabuf info from debugfs"; return false; } char* line = nullptr; size_t len = 0; dmabufs->clear(); while (getline(&line, &len, fp.get()) > 0) { // The new dmabuf bufinfo format adds inode number and a name at the end // We are looking for lines as follows: // size flags mode count exp_name ino name // 01048576 00000002 00000007 00000001 ion 00018758 CAMERA // 01048576 00000002 00000007 00000001 ion 00018758 uint64_t size, count, inode; char* exporter_name = nullptr; char* name = nullptr; int matched = sscanf(line, "%" SCNu64 "%*x %*x %" SCNu64 " %ms %" SCNu64 " %ms", &size, &count, &exporter_name, &inode, &name); if (matched < 4) { continue; } dmabufs->emplace_back((ino_t)inode, size, count, exporter_name, matched > 4 ? name : ""); free(exporter_name); free(name); } free(line); return true; } bool ReadDmaBufInfo(pid_t pid, std::vector* dmabufs, bool read_fdrefs) { dmabufs->clear(); return AppendDmaBufInfo(pid, dmabufs, read_fdrefs); } bool AppendDmaBufInfo(pid_t pid, std::vector* dmabufs, bool read_fdrefs) { if (read_fdrefs) { if (!ReadDmaBufFdRefs(pid, dmabufs)) { LOG(ERROR) << "Failed to read dmabuf fd references"; return false; } } if (!ReadDmaBufMapRefs(pid, dmabufs)) { LOG(ERROR) << "Failed to read dmabuf map references"; return false; } return true; } } // namespace dmabufinfo } // namespace android