Skip to content

Commit eb33144

Browse files
authored
Feature/serialization debug info (shader-slang#767)
* Remove AppContext. Use StdChannels to hold writers, and TestToolUtil to hold test tool specific functionality. * StdChannels -> StdWriters * getStdOut -> getOut, getStdError -> getError * Renamed main.cpp files of tools to try and stop visual studio getting confused between files - such that clicking on an error takes editor to the right location. * Work in progress on being able to serialize debug information. * * Added MemoryStream * First pass converting to IRSerialData * Able to read and write IRSerialData with debug data * Start at reconstruting IR serialized data. * First pass of generation debug SourceLocs from debug data. Works for test set for line nos. * Bug fixes. Moved testing of serialization into IRSerialUtil * Work around problem with irModule = generateIRForTranslationUnit(translationUnit); two times in a row produces different output(!). Fix by just creating once. * Remove problem with use of ternary op in slang.cpp on gcc/clang. * Added -verify-debug-serial-ir option that makes IR modules go through full serialization with debug information and verification. * Add a test that does serial debug verification that is run by default on linux.
1 parent d155eaa commit eb33144

15 files changed

+1070
-121
lines changed

source/core/slang-string-slice-pool.h

+2
Original file line numberDiff line numberDiff line change
@@ -50,6 +50,8 @@ class StringSlicePool
5050

5151
/// Convert a handle to and index. (A handle is just an index!)
5252
static int asIndex(Handle handle) { return int(handle); }
53+
/// Returns true if the handle is to a slice that contains characters (ie not null or empty)
54+
static bool hasContents(Handle handle) { return int(handle) >= kNumDefaultHandles; }
5355

5456
/// Ctor
5557
StringSlicePool();

source/core/stream.cpp

+76
Original file line numberDiff line numberDiff line change
@@ -153,6 +153,7 @@ namespace Slang
153153
break;
154154
case Slang::SeekOrigin::End:
155155
_origin = SEEK_END;
156+
// JS TODO: This doesn't seem right, the offset can mean it's not at the end
156157
endReached = true;
157158
break;
158159
case Slang::SeekOrigin::Current:
@@ -215,4 +216,79 @@ namespace Slang
215216
{
216217
return endReached;
217218
}
219+
220+
// !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! MemoryStream !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
221+
222+
void MemoryStream::Seek(SeekOrigin origin, Int64 offset)
223+
{
224+
Int64 pos = 0;
225+
switch (origin)
226+
{
227+
case Slang::SeekOrigin::Start:
228+
pos = offset;
229+
break;
230+
case Slang::SeekOrigin::End:
231+
pos = Int64(m_contents.Count()) + offset;
232+
break;
233+
case Slang::SeekOrigin::Current:
234+
pos = Int64(m_position) + offset;
235+
break;
236+
default:
237+
throw NotSupportedException("Unsupported seek origin.");
238+
break;
239+
}
240+
241+
m_atEnd = false;
242+
243+
// Clamp to the valid range
244+
pos = (pos < 0) ? 0 : pos;
245+
pos = (pos > Int64(m_contents.Count())) ? Int64(m_contents.Count()) : pos;
246+
247+
m_position = UInt(pos);
248+
}
249+
250+
Int64 MemoryStream::Read(void * buffer, Int64 length)
251+
{
252+
if (!CanRead())
253+
{
254+
throw IOException("Cannot read this stream.");
255+
}
256+
257+
const Int64 maxRead = Int64(m_contents.Count() - m_position);
258+
259+
if (maxRead == 0 && length > 0)
260+
{
261+
m_atEnd = true;
262+
throw EndOfStreamException("End of file is reached.");
263+
}
264+
265+
length = length > maxRead ? maxRead : length;
266+
267+
::memcpy(buffer, m_contents.begin() + m_position, size_t(length));
268+
m_position += UInt(length);
269+
return maxRead;
270+
}
271+
272+
Int64 MemoryStream::Write(const void * buffer, Int64 length)
273+
{
274+
if (!CanWrite())
275+
{
276+
throw IOException("Cannot write this stream.");
277+
}
278+
279+
if (m_position == m_contents.Count())
280+
{
281+
m_contents.AddRange((const uint8_t*)buffer, UInt(length));
282+
}
283+
else
284+
{
285+
m_contents.InsertRange(m_position, (const uint8_t*)buffer, UInt(length));
286+
}
287+
288+
m_atEnd = false;
289+
290+
m_position += UInt(length);
291+
return length;
292+
}
293+
218294
}

source/core/stream.h

+27-1
Original file line numberDiff line numberDiff line change
@@ -53,14 +53,40 @@ namespace Slang
5353

5454
enum class FileAccess
5555
{
56-
Read = 1, Write = 2, ReadWrite = 3
56+
None = 0, Read = 1, Write = 2, ReadWrite = 3
5757
};
5858

5959
enum class FileShare
6060
{
6161
None, ReadOnly, WriteOnly, ReadWrite
6262
};
6363

64+
class MemoryStream : public Stream
65+
{
66+
public:
67+
virtual Int64 GetPosition() SLANG_OVERRIDE { return m_position; }
68+
virtual void Seek(SeekOrigin origin, Int64 offset) SLANG_OVERRIDE;
69+
virtual Int64 Read(void * buffer, Int64 length) SLANG_OVERRIDE;
70+
virtual Int64 Write(const void * buffer, Int64 length) SLANG_OVERRIDE;
71+
virtual bool IsEnd() SLANG_OVERRIDE { return m_atEnd; }
72+
virtual bool CanRead() SLANG_OVERRIDE { return (int(m_access) & int(FileAccess::Read)) != 0; }
73+
virtual bool CanWrite() SLANG_OVERRIDE { return (int(m_access) & int(FileAccess::Write)) != 0; }
74+
virtual void Close() SLANG_OVERRIDE { m_access = FileAccess::None; }
75+
76+
MemoryStream(FileAccess access) :
77+
m_access(access),
78+
m_position(0),
79+
m_atEnd(false)
80+
{}
81+
82+
UInt m_position;
83+
84+
bool m_atEnd; ///< Happens when a read is done and nothing can be returned because already at end
85+
86+
FileAccess m_access;
87+
List<uint8_t> m_contents;
88+
};
89+
6490
class FileStream : public Stream
6591
{
6692
private:

source/slang/compiler.cpp

+3-3
Original file line numberDiff line numberDiff line change
@@ -260,7 +260,7 @@ namespace Slang
260260
{
261261
codeBuilder << "#line 1 \"";
262262

263-
const String& path = sourceFile->pathInfo.foundPath;
263+
const String& path = sourceFile->getPathInfo().foundPath;
264264

265265
for(auto c : path)
266266
{
@@ -277,7 +277,7 @@ namespace Slang
277277
}
278278
codeBuilder << "\"\n";
279279

280-
codeBuilder << sourceFile->content << "\n";
280+
codeBuilder << sourceFile->getContent() << "\n";
281281
}
282282

283283
return codeBuilder.ProduceString();
@@ -322,7 +322,7 @@ namespace Slang
322322
{
323323
codeBuilder << "#line 1 " << translationUnitIndex << "\n";
324324
}
325-
codeBuilder << sourceFile->content << "\n";
325+
codeBuilder << sourceFile->getContent() << "\n";
326326
}
327327

328328
return codeBuilder.ProduceString();

source/slang/compiler.h

+3
Original file line numberDiff line numberDiff line change
@@ -361,6 +361,9 @@ namespace Slang
361361
// serialization a bottleneck or firewall between the front end and the backend
362362
bool useSerialIRBottleneck = false;
363363

364+
// If true will serialize and de-serialize with debug information
365+
bool verifyDebugSerialization = false;
366+
364367
// How should `#line` directives be emitted (if at all)?
365368
LineDirectiveMode lineDirectiveMode = LineDirectiveMode::Default;
366369

source/slang/diagnostic-defs.h

+1
Original file line numberDiff line numberDiff line change
@@ -446,5 +446,6 @@ DIAGNOSTIC(99999, Internal, internalCompilerError, "Slang internal compiler erro
446446
DIAGNOSTIC(99999, Error, compilationAborted, "Slang compilation aborted due to internal error");
447447
DIAGNOSTIC(99999, Error, compilationAbortedDueToException, "Slang compilation aborted due to an exception of $0: $1");
448448
DIAGNOSTIC(99999, Note, noteLocationOfInternalError, "the Slang compiler threw an exception while working on code near this location");
449+
DIAGNOSTIC(99999, Internal, serialDebugVerificationFailed, "Verification of serial debug information failed.");
449450

450451
#undef DIAGNOSTIC

0 commit comments

Comments
 (0)