From 080493c058f52a5c20638f1b975d032af45f4d3f Mon Sep 17 00:00:00 2001 From: bptato Date: Thu, 19 Sep 2024 17:46:27 +0200 Subject: loader: mmap intermediate image files, misc refactoring * refactor parseHeader * optimize response blob() * add direct "to cache" mode for loader requests which sets stdout to a file, and use it for image processing * move image resizing into a separate process * mmap cache files in between processing steps when possible At last, resize is no longer a part of image decoding. Also, it feels much nicer to keep encoded image data in the same cache as everything else. The mmap operations *should* be more efficient than copying the whole RGBA data through a pipe. In practice, it only makes a difference for loading (well, now just mmapping) the encoded image into the pager, where it singlehandedly speeds up image display by 10x on my test image. For the other steps, the unfortunate fact that "tocache" must delay the next fork/exec in the pipeline until the entire image is processed seems to equal out any wins we might have gotten from skipping a single raw RGBA copy. I have tried moving the delay before the exec (it's possible with yet another pipe), but it didn't help much and made the code much uglier. (Not that tocache didn't, but I can live with this...) --- src/loader/response.nim | 35 +++++++++++++++++++---------------- 1 file changed, 19 insertions(+), 16 deletions(-) (limited to 'src/loader/response.nim') diff --git a/src/loader/response.nim b/src/loader/response.nim index 1d7307cd..23c386ef 100644 --- a/src/loader/response.nim +++ b/src/loader/response.nim @@ -47,6 +47,8 @@ type opaque*: RootRef flags*: set[ResponseFlag] + FetchPromise* = Promise[JSResult[Response]] + jsDestructor(Response) proc newResponse*(res: int; request: Request; stream: SocketStream; @@ -59,6 +61,9 @@ proc newResponse*(res: int; request: Request; stream: SocketStream; status: status ) +proc newFetchPromise*(): FetchPromise = + return newPromise[JSResult[Response]]() + func makeNetworkError*(): Response {.jsstfunc: "Response.error".} = #TODO use "create" function return Response( @@ -115,6 +120,10 @@ func getReferrerPolicy*(this: Response): Option[ReferrerPolicy] = this.headers.table.withValue("Referrer-Policy", p): return strictParseEnum[ReferrerPolicy](p[][0]) +proc resume*(response: Response) = + response.resumeFun(response.outputId) + response.resumeFun = nil + type TextOpaque = ref object of RootObj buf: string bodyRead: Promise[JSResult[string]] @@ -145,15 +154,9 @@ proc onFinishText(response: Response; success: bool) = let err = newTypeError("NetworkError when attempting to fetch resource") bodyRead.resolve(JSResult[string].err(err)) -proc resume*(response: Response) = - response.resumeFun(response.outputId) - response.resumeFun = nil - proc text*(response: Response): Promise[JSResult[string]] {.jsfunc.} = if response.body == nil: - let p = newPromise[JSResult[string]]() - p.resolve(JSResult[string].ok("")) - return p + return newResolvedPromise(JSResult[string].ok("")) if response.bodyUsed: let p = newPromise[JSResult[string]]() let err = JSResult[string] @@ -179,12 +182,12 @@ proc onReadBlob(response: Response) = let opaque = BlobOpaque(response.opaque) while true: try: - let targetLen = opaque.len + BufferSize - if targetLen > opaque.size: - opaque.size = targetLen - opaque.p = realloc(opaque.p, targetLen) + if opaque.len + BufferSize > opaque.size: + opaque.size *= 2 + opaque.p = realloc(opaque.p, opaque.size) let p = cast[ptr UncheckedArray[uint8]](opaque.p) - let n = response.body.recvData(addr p[opaque.len], BufferSize) + let diff = opaque.size - opaque.len + let n = response.body.recvData(addr p[opaque.len], diff) opaque.len += n if n == 0: break @@ -211,13 +214,13 @@ proc onFinishBlob(response: Response; success: bool) = proc blob*(response: Response): Promise[JSResult[Blob]] {.jsfunc.} = if response.bodyUsed: - let p = newPromise[JSResult[Blob]]() let err = JSResult[Blob].err(newTypeError("Body has already been consumed")) - p.resolve(err) - return p + return newResolvedPromise(err) let opaque = BlobOpaque( bodyRead: newPromise[JSResult[Blob]](), - contentType: response.getContentType() + contentType: response.getContentType(), + p: alloc(BufferSize), + size: BufferSize ) response.opaque = opaque response.onRead = onReadBlob -- cgit 1.4.1-2-gfad0