source_buffer.cpp 3.3 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889
  1. // Part of the Carbon Language project, under the Apache License v2.0 with LLVM
  2. // Exceptions. See /LICENSE for license information.
  3. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
  4. #include "toolchain/source/source_buffer.h"
  5. #include <limits>
  6. #include <memory>
  7. #include <optional>
  8. #include "llvm/Support/ErrorOr.h"
  9. #include "toolchain/diagnostics/file_diagnostics.h"
  10. namespace Carbon {
  11. auto SourceBuffer::MakeFromStdin(Diagnostics::Consumer& consumer)
  12. -> std::optional<SourceBuffer> {
  13. return MakeFromMemoryBuffer(llvm::MemoryBuffer::getSTDIN(), "<stdin>",
  14. /*is_regular_file=*/false, consumer);
  15. }
  16. auto SourceBuffer::MakeFromFile(llvm::vfs::FileSystem& fs,
  17. llvm::StringRef filename,
  18. Diagnostics::Consumer& consumer)
  19. -> std::optional<SourceBuffer> {
  20. Diagnostics::FileEmitter emitter(&consumer);
  21. llvm::ErrorOr<std::unique_ptr<llvm::vfs::File>> file =
  22. fs.openFileForRead(filename);
  23. if (file.getError()) {
  24. CARBON_DIAGNOSTIC(ErrorOpeningFile, Error,
  25. "error opening file for read: {0}", std::string);
  26. emitter.Emit(filename, ErrorOpeningFile, file.getError().message());
  27. return std::nullopt;
  28. }
  29. llvm::ErrorOr<llvm::vfs::Status> status = (*file)->status();
  30. if (status.getError()) {
  31. CARBON_DIAGNOSTIC(ErrorStattingFile, Error, "error statting file: {0}",
  32. std::string);
  33. emitter.Emit(filename, ErrorStattingFile, file.getError().message());
  34. return std::nullopt;
  35. }
  36. // `stat` on a file without a known size gives a size of 0, which causes
  37. // `llvm::vfs::File::getBuffer` to produce an empty buffer. Use a size of -1
  38. // in this case so we get the complete file contents.
  39. bool is_regular_file = status->isRegularFile();
  40. int64_t size = is_regular_file ? status->getSize() : -1;
  41. return MakeFromMemoryBuffer(
  42. (*file)->getBuffer(filename, size, /*RequiresNullTerminator=*/false),
  43. filename, is_regular_file, consumer);
  44. }
  45. auto SourceBuffer::MakeFromStringCopy(llvm::StringRef filename,
  46. llvm::StringRef text,
  47. Diagnostics::Consumer& consumer)
  48. -> std::optional<SourceBuffer> {
  49. return MakeFromMemoryBuffer(
  50. llvm::MemoryBuffer::getMemBufferCopy(text, filename), filename,
  51. /*is_regular_file=*/true, consumer);
  52. }
  53. auto SourceBuffer::MakeFromMemoryBuffer(
  54. llvm::ErrorOr<std::unique_ptr<llvm::MemoryBuffer>> buffer,
  55. llvm::StringRef filename, bool is_regular_file,
  56. Diagnostics::Consumer& consumer) -> std::optional<SourceBuffer> {
  57. Diagnostics::FileEmitter emitter(&consumer);
  58. if (buffer.getError()) {
  59. CARBON_DIAGNOSTIC(ErrorReadingFile, Error, "error reading file: {0}",
  60. std::string);
  61. emitter.Emit(filename, ErrorReadingFile, buffer.getError().message());
  62. return std::nullopt;
  63. }
  64. if (buffer.get()->getBufferSize() >= std::numeric_limits<int32_t>::max()) {
  65. CARBON_DIAGNOSTIC(FileTooLarge, Error,
  66. "file is over the 2GiB input limit; size is {0} bytes",
  67. int64_t);
  68. emitter.Emit(filename, FileTooLarge, buffer.get()->getBufferSize());
  69. return std::nullopt;
  70. }
  71. return SourceBuffer(filename.str(), std::move(buffer.get()), is_regular_file);
  72. }
  73. } // namespace Carbon