Guido van Rossum | 5dafd91 | 1991-11-04 18:04:21 +0000 | [diff] [blame] | 1 | import getopt |
Guido van Rossum | e4bddea | 1991-10-30 11:52:48 +0000 | [diff] [blame] | 2 | from gl import * |
| 3 | from GL import * |
| 4 | from DEVICE import * |
| 5 | import time |
| 6 | import sys |
| 7 | import al |
| 8 | import AL |
Guido van Rossum | 4476670 | 1992-05-12 12:33:44 +0000 | [diff] [blame] | 9 | |
| 10 | sys.path.append('/ufs/guido/src/video') # Increase chance to find colorsys |
Guido van Rossum | a63f197 | 1991-11-22 14:04:01 +0000 | [diff] [blame] | 11 | import colorsys |
Guido van Rossum | e4bddea | 1991-10-30 11:52:48 +0000 | [diff] [blame] | 12 | |
| 13 | BUFFERSIZE = 32000 |
| 14 | |
| 15 | class Struct(): pass |
| 16 | epoch = Struct() |
Guido van Rossum | a63f197 | 1991-11-22 14:04:01 +0000 | [diff] [blame] | 17 | epoch.correcttiming = 1 |
Guido van Rossum | e4bddea | 1991-10-30 11:52:48 +0000 | [diff] [blame] | 18 | EndOfFile = 'End of file' |
| 19 | bye = 'bye' |
| 20 | |
| 21 | def openspkr(): |
| 22 | conf = al.newconfig() |
| 23 | conf.setqueuesize(BUFFERSIZE) |
| 24 | conf.setwidth(AL.SAMPLE_16) |
| 25 | conf.setchannels(AL.MONO) |
| 26 | return al.openport('spkr','w',conf) |
Guido van Rossum | b51afcc | 1991-11-04 14:29:27 +0000 | [diff] [blame] | 27 | |
Guido van Rossum | e4bddea | 1991-10-30 11:52:48 +0000 | [diff] [blame] | 28 | def openvideo(name): |
Guido van Rossum | b51afcc | 1991-11-04 14:29:27 +0000 | [diff] [blame] | 29 | try: |
| 30 | f = open(name, 'r') |
| 31 | except: |
| 32 | sys.stderr.write(name + ': cannot open\n') |
| 33 | sys.exit(1) |
Guido van Rossum | e4bddea | 1991-10-30 11:52:48 +0000 | [diff] [blame] | 34 | line = f.readline() |
| 35 | if not line: raise EndOfFile |
Guido van Rossum | 2c8bf9d | 1992-05-06 17:58:18 +0000 | [diff] [blame] | 36 | colorinfo = (8, 0, 0, 0) |
Guido van Rossum | f47d048 | 1992-02-11 14:50:22 +0000 | [diff] [blame] | 37 | if line[:4] == 'CMIF': |
| 38 | if line[:14] == 'CMIF video 2.0': |
Guido van Rossum | a63f197 | 1991-11-22 14:04:01 +0000 | [diff] [blame] | 39 | line = f.readline() |
| 40 | colorinfo = eval(line[:-1]) |
Guido van Rossum | a63f197 | 1991-11-22 14:04:01 +0000 | [diff] [blame] | 41 | line = f.readline() |
Guido van Rossum | e4bddea | 1991-10-30 11:52:48 +0000 | [diff] [blame] | 42 | x = eval(line[:-1]) |
Guido van Rossum | f47d048 | 1992-02-11 14:50:22 +0000 | [diff] [blame] | 43 | if len(x) == 3: w, h, pf = x |
Guido van Rossum | e4bddea | 1991-10-30 11:52:48 +0000 | [diff] [blame] | 44 | else: w, h = x; pf = 2 |
Guido van Rossum | 864cde2 | 1992-05-07 15:21:25 +0000 | [diff] [blame] | 45 | if pf and w/pf % 4 <> 0: |
| 46 | sys.stderr.write( \ |
| 47 | 'warning: stride not a multiple of 4 -- may not work on Indigo XS\n') |
Guido van Rossum | a63f197 | 1991-11-22 14:04:01 +0000 | [diff] [blame] | 48 | return f, w, h, pf, colorinfo |
Guido van Rossum | b51afcc | 1991-11-04 14:29:27 +0000 | [diff] [blame] | 49 | |
Guido van Rossum | a63f197 | 1991-11-22 14:04:01 +0000 | [diff] [blame] | 50 | def loadframe(f,w,h,pf,af,spkr, (ybits,ibits,qbits,chrompack),mf): |
Guido van Rossum | e4bddea | 1991-10-30 11:52:48 +0000 | [diff] [blame] | 51 | line = f.readline() |
Guido van Rossum | f47d048 | 1992-02-11 14:50:22 +0000 | [diff] [blame] | 52 | if line == '': |
Guido van Rossum | e4bddea | 1991-10-30 11:52:48 +0000 | [diff] [blame] | 53 | raise EndOfFile |
| 54 | x = eval(line[:-1]) |
Guido van Rossum | f47d048 | 1992-02-11 14:50:22 +0000 | [diff] [blame] | 55 | if type(x) == type(0) or type(x) == type(0.0): |
Guido van Rossum | e4bddea | 1991-10-30 11:52:48 +0000 | [diff] [blame] | 56 | tijd = x |
Guido van Rossum | f47d048 | 1992-02-11 14:50:22 +0000 | [diff] [blame] | 57 | if pf == 0: |
Guido van Rossum | e4bddea | 1991-10-30 11:52:48 +0000 | [diff] [blame] | 58 | size = w*h*4 |
| 59 | else: |
| 60 | size = (w/pf) * (h/pf) |
| 61 | else: |
| 62 | tijd, size = x |
| 63 | data = f.read(size) |
| 64 | if len(data) <> size: |
| 65 | raise EndOfFile |
| 66 | if pf: |
Guido van Rossum | e4bddea | 1991-10-30 11:52:48 +0000 | [diff] [blame] | 67 | w = w/pf |
| 68 | h = h/pf |
Guido van Rossum | a63f197 | 1991-11-22 14:04:01 +0000 | [diff] [blame] | 69 | if chrompack: |
| 70 | cw = (w+chrompack-1)/chrompack |
| 71 | ch = (h+chrompack-1)/chrompack |
| 72 | chromdata = f.read(2*cw*ch) |
| 73 | rectzoom(pf*chrompack*mf,pf*chrompack*mf) |
Guido van Rossum | 864cde2 | 1992-05-07 15:21:25 +0000 | [diff] [blame] | 74 | pixmode(PM_SIZE,16) |
Guido van Rossum | a63f197 | 1991-11-22 14:04:01 +0000 | [diff] [blame] | 75 | writemask(0x7ff - ((1<<ybits)-1)) |
| 76 | lrectwrite(0,0,cw-1,ch-1,chromdata) |
| 77 | writemask((1<<ybits)-1) |
Guido van Rossum | 864cde2 | 1992-05-07 15:21:25 +0000 | [diff] [blame] | 78 | pixmode(PM_SIZE,8) |
Guido van Rossum | a63f197 | 1991-11-22 14:04:01 +0000 | [diff] [blame] | 79 | if pf: |
| 80 | rectzoom(pf*mf, pf*mf) |
| 81 | elif mf <> 1: |
| 82 | rectzoom(mf,mf) |
Guido van Rossum | e4bddea | 1991-10-30 11:52:48 +0000 | [diff] [blame] | 83 | lrectwrite(0,0,w-1,h-1,data) |
| 84 | # This is ugly here, but the only way to get the two |
| 85 | # channels started in sync |
| 86 | #if af <> None: |
| 87 | # playsound(af,spkr) |
| 88 | ct = time.millitimer() - epoch.epoch |
Guido van Rossum | a63f197 | 1991-11-22 14:04:01 +0000 | [diff] [blame] | 89 | if epoch.correcttiming and tijd > 0 and ct < tijd: |
Guido van Rossum | e4bddea | 1991-10-30 11:52:48 +0000 | [diff] [blame] | 90 | time.millisleep(tijd-ct) |
Guido van Rossum | b51afcc | 1991-11-04 14:29:27 +0000 | [diff] [blame] | 91 | #swapbuffers() |
Guido van Rossum | e4bddea | 1991-10-30 11:52:48 +0000 | [diff] [blame] | 92 | return tijd |
Guido van Rossum | b51afcc | 1991-11-04 14:29:27 +0000 | [diff] [blame] | 93 | |
Guido van Rossum | a63f197 | 1991-11-22 14:04:01 +0000 | [diff] [blame] | 94 | def initcmap(ybits,ibits,qbits,chrompack): |
| 95 | if ybits+ibits+qbits > 11: |
| 96 | raise 'Sorry, 11 bits max' |
| 97 | maxy = pow(2,ybits) |
| 98 | maxi = pow(2,ibits) |
| 99 | maxq = pow(2,qbits) |
| 100 | for i in range(2048,4096-256): |
| 101 | mapcolor(i, 0, 255, 0) |
| 102 | for y in range(maxy): |
| 103 | yv = float(y)/float(maxy-1) |
| 104 | for i in range(maxi): |
Guido van Rossum | f47d048 | 1992-02-11 14:50:22 +0000 | [diff] [blame] | 105 | if maxi == 1: iv = 0 |
Guido van Rossum | 696f911 | 1991-12-03 17:25:52 +0000 | [diff] [blame] | 106 | else: iv = (float(i)/float(maxi-1))-0.5 |
Guido van Rossum | a63f197 | 1991-11-22 14:04:01 +0000 | [diff] [blame] | 107 | for q in range(maxq): |
Guido van Rossum | f47d048 | 1992-02-11 14:50:22 +0000 | [diff] [blame] | 108 | if maxq == 1: qv = 0 |
Guido van Rossum | 696f911 | 1991-12-03 17:25:52 +0000 | [diff] [blame] | 109 | else: qv = (float(q)/float(maxq-1))-0.5 |
Guido van Rossum | a63f197 | 1991-11-22 14:04:01 +0000 | [diff] [blame] | 110 | index = 2048 + y + (i << ybits) + (q << (ybits+ibits)) |
Guido van Rossum | a63f197 | 1991-11-22 14:04:01 +0000 | [diff] [blame] | 111 | rv,gv,bv = colorsys.yiq_to_rgb(yv,iv,qv) |
| 112 | r,g,b = int(rv*255.0), int(gv*255.0), int(bv*255.0) |
| 113 | if index < 4096 - 256: |
| 114 | mapcolor(index, r,g,b) |
| 115 | |
Guido van Rossum | e4bddea | 1991-10-30 11:52:48 +0000 | [diff] [blame] | 116 | def playsound(af, spkr): |
| 117 | nsamp = spkr.getfillable() |
| 118 | data = af.read(nsamp*2) |
| 119 | spkr.writesamps(data) |
Guido van Rossum | b51afcc | 1991-11-04 14:29:27 +0000 | [diff] [blame] | 120 | |
Guido van Rossum | e4bddea | 1991-10-30 11:52:48 +0000 | [diff] [blame] | 121 | def main(): |
Guido van Rossum | 5dafd91 | 1991-11-04 18:04:21 +0000 | [diff] [blame] | 122 | looping = 0 |
| 123 | packfactor = 0 |
Guido van Rossum | a63f197 | 1991-11-22 14:04:01 +0000 | [diff] [blame] | 124 | magfactor = 1 |
| 125 | try: |
| 126 | opts, args = getopt.getopt(sys.argv[1:], 'm:p:lF') |
| 127 | except getopt.error: |
| 128 | sys.stderr.write('usage: video ' + \ |
| 129 | '[-l] [-p pf] [-m mag] [-F] [moviefile [soundfile [skipbytes]]]\n') |
| 130 | sys.exit(2) |
Guido van Rossum | 5dafd91 | 1991-11-04 18:04:21 +0000 | [diff] [blame] | 131 | for opt, arg in opts: |
Guido van Rossum | f47d048 | 1992-02-11 14:50:22 +0000 | [diff] [blame] | 132 | if opt == '-m': |
Guido van Rossum | a63f197 | 1991-11-22 14:04:01 +0000 | [diff] [blame] | 133 | magfactor = int(eval(arg)) |
Guido van Rossum | f47d048 | 1992-02-11 14:50:22 +0000 | [diff] [blame] | 134 | elif opt == '-p': |
Guido van Rossum | 5dafd91 | 1991-11-04 18:04:21 +0000 | [diff] [blame] | 135 | packfactor = int(eval(arg)) |
Guido van Rossum | f47d048 | 1992-02-11 14:50:22 +0000 | [diff] [blame] | 136 | elif opt == '-l': |
Guido van Rossum | 5dafd91 | 1991-11-04 18:04:21 +0000 | [diff] [blame] | 137 | looping = 1 |
Guido van Rossum | f47d048 | 1992-02-11 14:50:22 +0000 | [diff] [blame] | 138 | elif opt == '-F': |
Guido van Rossum | a63f197 | 1991-11-22 14:04:01 +0000 | [diff] [blame] | 139 | epoch.correcttiming = 0 |
Guido van Rossum | 5dafd91 | 1991-11-04 18:04:21 +0000 | [diff] [blame] | 140 | if args: |
| 141 | filename = args[0] |
Guido van Rossum | b51afcc | 1991-11-04 14:29:27 +0000 | [diff] [blame] | 142 | else: |
| 143 | filename = 'film.video' |
Guido van Rossum | a63f197 | 1991-11-22 14:04:01 +0000 | [diff] [blame] | 144 | f, w, h, pf, cinfo = openvideo(filename) |
Guido van Rossum | 5dafd91 | 1991-11-04 18:04:21 +0000 | [diff] [blame] | 145 | if 0 < packfactor <> pf: |
| 146 | w = w/pf*packfactor |
| 147 | h = h/pf*packfactor |
| 148 | pf = packfactor |
| 149 | if args[1:]: |
| 150 | audiofilename = args[1] |
Guido van Rossum | b51afcc | 1991-11-04 14:29:27 +0000 | [diff] [blame] | 151 | af = open(audiofilename, 'r') |
| 152 | spkr = openspkr() |
Guido van Rossum | 5dafd91 | 1991-11-04 18:04:21 +0000 | [diff] [blame] | 153 | afskip = 0 |
Guido van Rossum | b3c493c | 1991-11-04 18:14:23 +0000 | [diff] [blame] | 154 | if args[2:]: |
| 155 | afskip = eval(args[2]) |
| 156 | af.seek(afskip) |
Guido van Rossum | b51afcc | 1991-11-04 14:29:27 +0000 | [diff] [blame] | 157 | else: |
| 158 | af, spkr = None, None |
Guido van Rossum | 5dafd91 | 1991-11-04 18:04:21 +0000 | [diff] [blame] | 159 | foreground() |
Guido van Rossum | a63f197 | 1991-11-22 14:04:01 +0000 | [diff] [blame] | 160 | prefsize(w*magfactor,h*magfactor) |
Guido van Rossum | b51afcc | 1991-11-04 14:29:27 +0000 | [diff] [blame] | 161 | win = winopen(filename) |
Guido van Rossum | a63f197 | 1991-11-22 14:04:01 +0000 | [diff] [blame] | 162 | if pf: |
| 163 | #doublebuffer() |
| 164 | cmode() |
| 165 | else: |
| 166 | RGBmode() |
Guido van Rossum | b51afcc | 1991-11-04 14:29:27 +0000 | [diff] [blame] | 167 | #doublebuffer() |
| 168 | gconfig() |
Guido van Rossum | a63f197 | 1991-11-22 14:04:01 +0000 | [diff] [blame] | 169 | if pf: |
| 170 | initcmap(cinfo) |
| 171 | color(2048) |
| 172 | clear() |
| 173 | writemask(2047) |
Guido van Rossum | 864cde2 | 1992-05-07 15:21:25 +0000 | [diff] [blame] | 174 | pixmode(PM_SIZE,8) # 8 bit pixels |
Guido van Rossum | b51afcc | 1991-11-04 14:29:27 +0000 | [diff] [blame] | 175 | qdevice(ESCKEY) |
| 176 | qdevice(WINSHUT) |
| 177 | qdevice(WINQUIT) |
| 178 | running = 1 |
| 179 | epoch.epoch = time.millitimer() |
| 180 | nframe = 0 |
| 181 | tijd = 1 |
Guido van Rossum | 5dafd91 | 1991-11-04 18:04:21 +0000 | [diff] [blame] | 182 | if looping: |
| 183 | looping = f.tell() |
Guido van Rossum | b51afcc | 1991-11-04 14:29:27 +0000 | [diff] [blame] | 184 | try: |
| 185 | while 1: |
| 186 | if running: |
| 187 | try: |
Guido van Rossum | a63f197 | 1991-11-22 14:04:01 +0000 | [diff] [blame] | 188 | tijd = loadframe(f, w, h, pf, af, spkr, cinfo,magfactor) |
Guido van Rossum | b51afcc | 1991-11-04 14:29:27 +0000 | [diff] [blame] | 189 | nframe = nframe + 1 |
| 190 | except EndOfFile: |
| 191 | running = 0 |
| 192 | t = time.millitimer() |
| 193 | if tijd > 0: |
| 194 | print 'Recorded at', |
| 195 | print 0.1 * int(nframe * 10000.0 / tijd), |
| 196 | print 'frames/sec' |
| 197 | print 'Played', nframe, 'frames at', |
| 198 | print 0.1 * int(nframe * 10000.0 / (t-epoch.epoch)), |
| 199 | print 'frames/sec' |
Guido van Rossum | 5dafd91 | 1991-11-04 18:04:21 +0000 | [diff] [blame] | 200 | if looping: |
| 201 | f.seek(looping) |
| 202 | epoch.epoch = time.millitimer() |
| 203 | nframe = 0 |
| 204 | running = 1 |
| 205 | if af <> None: |
| 206 | af.seek(afskip) |
Guido van Rossum | b51afcc | 1991-11-04 14:29:27 +0000 | [diff] [blame] | 207 | if af <> None: |
| 208 | playsound(af,spkr) |
| 209 | if not running or qtest(): |
| 210 | dev, val = qread() |
| 211 | if dev in (ESCKEY, WINSHUT, WINQUIT): |
| 212 | raise bye |
Guido van Rossum | f47d048 | 1992-02-11 14:50:22 +0000 | [diff] [blame] | 213 | elif dev == REDRAW: |
Guido van Rossum | b51afcc | 1991-11-04 14:29:27 +0000 | [diff] [blame] | 214 | reshapeviewport() |
| 215 | except bye: |
| 216 | pass |
Guido van Rossum | e4bddea | 1991-10-30 11:52:48 +0000 | [diff] [blame] | 217 | |
| 218 | main() |