racket/collects/unstable/srcloc.ss
2009-11-25 20:49:29 +00:00

322 lines
9.5 KiB
Scheme

#lang scheme/base
;; Unstable library by: Carl Eastlund <cce@ccs.neu.edu>
;; intended for use in scheme/contract, so don't try to add contracts!
(provide
;; type predicates
source-location?
source-location-list?
source-location-vector?
;; error checks
check-source-location!
;; conversion and combination
build-source-location
build-source-location-list
build-source-location-vector
build-source-location-syntax
;; accessors
source-location-known?
source-location-source
source-location-line
source-location-column
source-location-position
source-location-span
source-location-end
;; rendering
source-location->string
source-location->prefix
)
(define (source-location? x)
(process-source-location x good? bad? 'source-location?))
(define (source-location-list? x)
(process-list x good? bad? 'source-location-list?))
(define (source-location-vector? x)
(process-vector x good? bad? 'source-location-vector?))
(define (check-source-location! name x)
(process-source-location x good! bad! name))
(define (source-location-known? x)
(process-source-location x good-known? bad! 'source-location-known?))
(define (source-location-source x)
(process-source-location x good-source bad! 'source-location-source))
(define (source-location-line x)
(process-source-location x good-line bad! 'source-location-line))
(define (source-location-position x)
(process-source-location x good-position bad! 'source-location-position))
(define (source-location-column x)
(process-source-location x good-column bad! 'source-location-column))
(define (source-location-span x)
(process-source-location x good-span bad! 'source-location-span))
(define (source-location-end x)
(process-source-location x good-end bad! 'source-location-end))
(define (source-location->string x)
(process-source-location x good-string bad! 'source-location->string))
(define (source-location->prefix x)
(process-source-location x good-prefix bad! 'source-location->prefix))
(define (build-source-location . locs)
(combine-source-locations locs good-srcloc bad!
'build-source-location))
(define (build-source-location-list . locs)
(combine-source-locations locs good-list bad!
'build-source-location-list))
(define (build-source-location-vector . locs)
(combine-source-locations locs good-vector bad!
'build-source-location-vector))
(define (build-source-location-syntax . locs)
(combine-source-locations locs good-syntax bad!
'build-source-location-syntax))
(define (good? x src line col pos span) #t)
(define (bad? fmt . args) #f)
(define (good! x src line col pos span) (void))
(define (bad! fmt . args)
(raise
(make-exn:fail:contract
(apply format fmt args)
(current-continuation-marks))))
(define (good-known? x src line col pos span)
(and (or src line col pos span) #t))
(define (good-source x src line col pos span) src)
(define (good-line x src line col pos span) line)
(define (good-column x src line col pos span) col)
(define (good-position x src line col pos span) pos)
(define (good-span x src line col pos span) span)
(define (good-end x src line col pos span) (and pos span (+ pos span)))
(define (good-srcloc x src line col pos span)
(if (srcloc? x) x (make-srcloc src line col pos span)))
(define (good-list x src line col pos span)
(if (list? x) x (list src line col pos span)))
(define (good-vector x src line col pos span)
(if (vector? x) x (vector src line col pos span)))
(define (good-syntax x src line col pos span)
(cond
[(syntax? x) x]
[(or (list? x) (vector? x)) (datum->syntax #f null x)]
[else (datum->syntax #f null (vector src line col pos span))]))
(define (good-string x src line col pos span)
(format "~a~a"
(or src "")
(if line
(if col
(format ":~a.~a" line col)
(format ":~a" line))
(if pos
(if span
(format "::~a-~a" pos (+ pos span))
(format "::~a" pos))
""))))
(define (good-prefix x src line col pos span)
(let ([str (good-string x src line col pos span)])
(if (string=? str "") "" (string-append str ": "))))
(define (combine-source-locations locs good bad name)
(define (loop loc1 src1 line1 col1 pos1 span1 locs)
(if (null? locs)
(good loc1 src1 line1 col1 pos1 span1)
(process-source-location
(car locs)
(lambda (loc2 src2 line2 col2 pos2 span2)
(combine-two
src1 line1 col1 pos1 span1
src2 line2 col2 pos2 span2
(lambda (loc src line col pos span)
(loop loc src line col pos span (cdr locs)))))
bad
name)))
(if (null? locs)
(process-source-location #f good bad name)
(process-source-location
(car locs)
(lambda (loc src line col pos span)
(loop loc src line col pos span (cdr locs)))
bad
name)))
(define (combine-two src1 line1 col1 pos1 span1
src2 line2 col2 pos2 span2
good)
(if (and src1 src2 (equal? src1 src2))
(let-values
([(src) src1]
[(line col)
(cond
[(and line1 line2)
(cond
[(< line1 line2) (values line1 col1)]
[(> line1 line2) (values line2 col2)]
[else (values line1
(if (and col1 col2)
(min col1 col2)
(or col1 col2)))])]
[line1 (values line1 col1)]
[line2 (values line2 col2)]
[else (values #f #f)])]
[(pos span)
(cond
[(and pos1 pos2)
(let ([pos (min pos1 pos2)])
(cond
[(and span1 span2)
(let ([end (max (+ pos1 span1) (+ pos2 span2))])
(values pos (- end pos)))]
[span1 (values pos (- (+ pos1 span1) pos))]
[span2 (values pos (- (+ pos2 span2) pos))]
[else (values pos #f)]))])])
(good #f src line col pos span))
(good #f #f #f #f #f #f)))
(define (process-source-location x good bad name)
(cond
;; #f
[(not x) (process-false x good bad name)]
;; srcloc
[(srcloc? x) (process-srcloc x good bad name)]
;; list
[(or (null? x) (pair? x)) (process-list x good bad name)]
;; vector
[(vector? x) (process-vector x good bad name)]
;; syntax
[(syntax? x) (process-syntax x good bad name)]
;; other
[else
(bad
"~a: expected a source location (srcloc struct, syntax object, list, vector, or #f); got: ~e"
name
x)]))
(define (process-false x good bad name)
(process-elements #f good bad name #f #f #f #f #f))
(define (process-srcloc x good bad name)
(process-elements x good bad name
(srcloc-source x)
(srcloc-line x)
(srcloc-column x)
(srcloc-position x)
(srcloc-span x)))
(define (process-syntax x good bad name)
(process-elements x good bad name
(syntax-source x)
(syntax-line x)
(syntax-column x)
(syntax-position x)
(syntax-span x)))
(define (process-list x good bad name)
(cond
[(null? x)
(bad
"~a: expected a source location (a list of 5 elements); got an empty list: ~e"
name
x)]
[(list? x)
(let ([n (length x)])
(if (= n 5)
(apply process-elements x good bad name x)
(bad
"~a: expected a source location (a list of 5 elements); got a list of ~a elements: ~e"
name
n
x)))]
[(pair? x)
(bad
"~a: expected a source location (a list of 5 elements); got an improper list: ~e"
name
x)]
[else
(bad
"~a: expected a source location list; got: ~e"
name
x)]))
(define (process-vector x good bad name)
(if (vector? x)
(let ([n (vector-length x)])
(if (= n 5)
(process-elements x good bad name
(vector-ref x 0)
(vector-ref x 1)
(vector-ref x 2)
(vector-ref x 3)
(vector-ref x 4))
(bad
"~a: expected a source location (a vector of 5 elements); got a vector of ~a elements: ~e"
name
n
x)))
(bad
"~a: expected a source location vector; got: ~e"
name
x)))
(define (process-elements x good bad name src line col pos span)
(cond
[(and line (not (exact-positive-integer? line)))
(bad
"~a: expected a source location with a positive line number or #f (second element); got line number ~e: ~e"
name
line
x)]
[(and col (not (exact-nonnegative-integer? col)))
(bad
"~a: expected a source location with a non-negative column number or #f (third element); got column number ~e: ~e"
name
col
x)]
[(or (and col (not line)) (and (not col) line))
(bad
"~a: expected a source location with line number and column number both numeric or both #f; got ~a and ~a respectively: ~e"
name
line
col
x)]
[(and pos (not (exact-positive-integer? pos)))
(bad
"~a: expected a source location with a positive position or #f (fourth element); got line number ~e: ~e"
name
pos
x)]
[(and span (not (exact-nonnegative-integer? span)))
(bad
"~a: expected a source location with a non-negative span or #f (fifth element); got column number ~e: ~e"
name
span
x)]
[else (good x src line col pos span)]))