## Copyright (c) 2007-2009 Nokia Corporation and/or its subsidiary(-ies).# All rights reserved.# This component and the accompanying materials are made available# under the terms of the License "Eclipse Public License v1.0"# which accompanies this distribution, and is available# at the URL "http://www.eclipse.org/legal/epl-v10.html".## Initial Contributors:# Nokia Corporation - initial contribution.## Contributors:## Description: # # Raptor log visualisation program. Takes a raptor log as standard input# and displays timelines that represent build progress and # how much actual parallelism there is in the build.# This program requires the pygame and PyOpenGL modules.from OpenGL.GL import *from OpenGL.GLU import *import pygamefrom pygame.locals import *import timeclass Timeline(object): """A bar representing a number of recipes which were executed in time sequence. There is no guarantee about what host but in theory they could have been executed on the same host.""" globalmax = 2.0 def __init__(self,ylevel): self.maxtime = 0.0 self.recipes = [] self.ylevel = ylevel def append(self, recipe): "" add this recipe to this timeline if it happens after the latest recipe already in the timeline "" if recipe.starttime + recipe.duration > self.maxtime: self.maxtime = recipe.starttime + recipe.duration if self.maxtime > Timeline.globalmax: Timeline.globalmax = self.maxtime else: pass self.recipes.append(recipe) def draw(self): glLoadIdentity() self.xscale = 4.0 / Timeline.globalmax glTranslatef(-2.0, -1.5, -6.0) count = 0 for r in self.recipes: if count % 2 == 0: coloff=0.8 else: coloff = 1.0 count += 1 r.draw(self.xscale, self.ylevel, coloff)class Recipe(object): """Represents a task completed in a raptor build. Drawn as a colour-coded bar with different colours for the various recipe types.""" STAT_OK = 0 colours = { 'compile': (0.5,0.5,1.0), 'compile2object': (0.5,0.5,1.0), 'win32compile2object': (0.5,0.5,1.0), 'tools2linkexe': (0.5,1.0,0.5), 'link': (0.5,1.0,0.5), 'linkandpostlink': (0.5,1.0,0.5), 'win32stageonelink': (0.5,1.0,0.5), 'tools2lib': (0.5,1.0,1.0), 'win32stagetwolink': (1.0,0.1,1.0), 'postlink': (1.0,0.5,1.0) } def __init__(self, starttime, duration, name, status): self.starttime = starttime self.duration = duration self.status = status self.colour = (1.0, 1.0, 1.0) if name in Recipe.colours: self.colour = Recipe.colours[name] else: self.colour = (1.0,1.0,1.0) self.name = name def draw(self, scale, ylevel, coloff): if self.status == Recipe.STAT_OK: glColor4f(self.colour[0]*coloff, self.colour[1]*coloff, self.colour[2]*coloff,0.2) else: glColor4f(1.0*coloff, 0.6*coloff, 0.6*coloff,0.2) x = self.starttime * scale y = ylevel x2 = x + self.duration * scale y2 = ylevel + 0.2 glBegin(GL_QUADS) glVertex3f(x, y, 0) glVertex3f(x, y2, 0) glVertex3f(x2, y2, 0) glVertex3f(x2, y, 0) glEnd()def resize((width, height)): if height==0: height=1 glViewport(0, 0, width, height) glMatrixMode(GL_PROJECTION) glLoadIdentity() gluPerspective(45, 1.0*width/height, 0.1, 100.0) glMatrixMode(GL_MODELVIEW) glLoadIdentity()def init(): glShadeModel(GL_SMOOTH) glClearColor(0.0, 0.0, 0.0, 0.0) glClearDepth(1.0) glEnable(GL_DEPTH_TEST) glDepthFunc(GL_LEQUAL) glHint(GL_PERSPECTIVE_CORRECTION_HINT, GL_NICEST)import sysimport redef main(): video_flags = OPENGL|DOUBLEBUF pygame.init() pygame.display.set_mode((800,600), video_flags) resize((800,600)) init() frames = 0 ticks = pygame.time.get_ticks() lines = 4 timelines = [] ylevel = 0.0 for i in xrange(0,4): ylevel += 0.6 timelines.append(Timeline(ylevel)) f = sys.stdin recipe_re = re.compile(".*<recipe name='([^']+)'.*") time_re = re.compile(".*<time start='([0-9]+\.[0-9]+)' *elapsed='([0-9]+\.[0-9]+)'.*") status_re = re.compile(".*<status exit='([^']*)'.*") alternating = 0 start_time = 0.0 for l in f.xreadlines(): l2 = l.rstrip("\n") rm = recipe_re.match(l2) if rm is not None: rname = rm.groups()[0] continue tm = time_re.match(l2) if tm is not None: s = float(tm.groups()[0]) elapsed = float(tm.groups()[1]) if start_time == 0.0: start_time = s s -= start_time continue sm = status_re.match(l2) if sm is None: continue if sm.groups()[0] == 'ok': status = 0 else: status = int(sm.groups()[0]) olddiff = 999999999.0 tnum = 0 for t in timelines: newdiff = s - t.maxtime if newdiff < 0.0: continue if olddiff > newdiff: dest_timeline = t olddiff = newdiff tnum += 1 dest_timeline.append(Recipe(s, elapsed, rname, status)) event = pygame.event.poll() if event.type == QUIT or (event.type == KEYDOWN and event.key == K_ESCAPE): break glClear(GL_COLOR_BUFFER_BIT|GL_DEPTH_BUFFER_BIT) for t in timelines: t.draw() pygame.display.flip() frames = frames+1 print "fps: %de" % ((frames*1000)/(pygame.time.get_ticks()-ticks)) event = pygame.event.wait()if __name__ == '__main__': main()