[Piglit] [PATCH 6/7] Add SyncFileWriter for write/close synchronization on results file.

U. Artie Eoff ullysses.a.eoff at intel.com
Mon Feb 7 18:55:00 PST 2011


Add SyncFileWriter class to synchronize writes to the 'main' results
file from multiple threads.  This helps to ensure that writes to this
file are not intermingled.
---
 framework/core.py |   11 +++++------
 piglit-run.py     |   40 +++++++++++++++++++++++++++++++++++-----
 2 files changed, 40 insertions(+), 11 deletions(-)

diff --git a/framework/core.py b/framework/core.py
index 0e0aca9..ce388aa 100644
--- a/framework/core.py
+++ b/framework/core.py
@@ -123,8 +123,7 @@ class TestResult(dict):
 			else:
 				print >> result, k + ": " + encode(str(v))
 		print >> result, "!"
-		print >> file, result.getvalue(),
-
+		file.write(result.getvalue())
 
 class GroupResult(dict):
 	def __init__(self, *args):
@@ -282,9 +281,9 @@ class Environment:
 
 	def collectData(self):
 		if platform.system() != 'Windows':
-			print >>self.file, "glxinfo:", '@@@' + encode(self.run('glxinfo'))
+			self.file.write("glxinfo:", '@@@' + encode(self.run('glxinfo')), "\n")
 		if platform.system() == 'Linux':
-			print >>self.file, "lspci:", '@@@' + encode(self.run('lspci'))
+			self.file.write("lspci:", '@@@' + encode(self.run('lspci')), "\n")
 
 
 class Test:
@@ -343,7 +342,7 @@ class Test:
 			if Test.sleep:
 				time.sleep(Test.sleep)
 		else:
-			print "Dry-run: %(path)s" % locals()
+			status("dry-run")
 
 	# Returns True iff the given error message should be ignored
 	def isIgnored(self, error):
@@ -389,7 +388,7 @@ class TestProfile:
 		self.tests.doRun(env, '')
 		ThreadPools().joinAll()
 		time_end = time.time()
-		print >>env.file, "time:",(time_end-time_start)
+		env.file.write("time:", time_end-time_start, "\n")
 
 #############################################################################
 ##### Loaders
diff --git a/piglit-run.py b/piglit-run.py
index 9f0eca6..1e6d515 100755
--- a/piglit-run.py
+++ b/piglit-run.py
@@ -24,11 +24,41 @@
 
 from getopt import getopt, GetoptError
 import re
-import sys
+import sys, os
 
 import framework.core as core
-
-
+from framework.threads import synchronized_self
+
+class SyncFileWriter:
+	'''
+		Using the 'print' syntax to write to an instance of this class
+		may have unexpected results in a multithreaded program.  For example:
+			print >> file, "a", "b", "c"
+		will call write() to write "a", then call write() to write "b", and so on...
+		This type of execution allows for another thread to call write() before
+		the original statement completes its execution.
+		To avoid this behavior, call file.write() explicitly.  For example:
+			file.write("a", "b", "c", "\n")
+		will ensure that "a b c" gets written to the file before any other thread
+		is given write access.
+	'''
+	def __init__(self, filename):
+		self.file = open(filename, 'w')
+
+	@synchronized_self
+	def write(self, *args):
+		[self.file.write(str(a)) for a in args]
+		self.file.flush()
+		os.fsync(self.file.fileno())
+
+	@synchronized_self
+	def writeLine(self, *args):
+		self.write(*args)
+		self.write('\n')
+
+	@synchronized_self
+	def close(self):
+		self.file.close()
 
 #############################################################################
 ##### Main program
@@ -91,8 +121,8 @@ def main():
 	core.checkDir(resultsDir, False)
 
 	profile = core.loadTestProfile(profileFilename)
-	env.file = open(resultsDir + '/main', "w")
-	print >>env.file, "name: %(name)s" % { 'name': core.encode(OptionName) }
+	env.file = SyncFileWriter(resultsDir + '/main')
+	env.file.writeLine("name: %(name)s" % { 'name': core.encode(OptionName) })
 	env.collectData()
 	profile.run(env)
 	env.file.close()
-- 
1.7.3.4



More information about the Piglit mailing list