| 
									
										
										
										
											2012-08-16 01:00:30 +01:00
										 |  |  | #include <assert.h>
 | 
					
						
							| 
									
										
										
										
											2013-11-30 02:05:42 -06:00
										 |  |  | #include <errno.h>
 | 
					
						
							|  |  |  | #include <stdarg.h>
 | 
					
						
							| 
									
										
										
										
											2012-08-16 01:00:30 +01:00
										 |  |  | #include <stdio.h>
 | 
					
						
							|  |  |  | #include <stdlib.h>
 | 
					
						
							|  |  |  | #include <stdint.h>
 | 
					
						
							| 
									
										
										
										
											2014-07-09 00:55:20 -04:00
										 |  |  | #include <sys/stat.h>
 | 
					
						
							| 
									
										
										
										
											2012-08-16 01:00:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-14 22:08:18 +01:00
										 |  |  | #include "exec_stack.h"
 | 
					
						
							| 
									
										
										
										
											2012-08-16 01:00:30 +01:00
										 |  |  | #include "bytecode.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-18 16:52:47 +00:00
										 |  |  | #include "jv_alloc.h"
 | 
					
						
							| 
									
										
										
										
											2013-02-03 00:09:47 -06:00
										 |  |  | #include "jq_parser.h"
 | 
					
						
							| 
									
										
										
										
											2012-09-18 22:17:13 +01:00
										 |  |  | #include "locfile.h"
 | 
					
						
							| 
									
										
										
										
											2012-08-27 10:11:55 +01:00
										 |  |  | #include "jv.h"
 | 
					
						
							| 
									
										
										
										
											2013-06-16 08:25:12 -05:00
										 |  |  | #include "jq.h"
 | 
					
						
							| 
									
										
										
										
											2012-09-18 22:17:13 +01:00
										 |  |  | #include "parser.h"
 | 
					
						
							|  |  |  | #include "builtin.h"
 | 
					
						
							| 
									
										
										
										
											2014-07-09 00:55:20 -04:00
										 |  |  | #include "util.h"
 | 
					
						
							|  |  |  | #include "linker.h"
 | 
					
						
							| 
									
										
										
										
											2012-08-16 01:00:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-28 18:33:45 -05:00
										 |  |  | struct jq_state { | 
					
						
							| 
									
										
										
										
											2013-06-06 17:26:15 -05:00
										 |  |  |   void (*nomem_handler)(void *); | 
					
						
							|  |  |  |   void *nomem_handler_data; | 
					
						
							|  |  |  |   struct bytecode* bc; | 
					
						
							| 
									
										
										
										
											2013-06-18 00:06:00 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-30 02:05:42 -06:00
										 |  |  |   jq_err_cb err_cb; | 
					
						
							|  |  |  |   void *err_cb_data; | 
					
						
							| 
									
										
										
										
											2014-07-05 20:54:42 -05:00
										 |  |  |   jv error; | 
					
						
							| 
									
										
										
										
											2013-11-30 02:05:42 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-13 21:50:32 +01:00
										 |  |  |   struct stack stk; | 
					
						
							|  |  |  |   stack_ptr curr_frame; | 
					
						
							|  |  |  |   stack_ptr stk_top; | 
					
						
							|  |  |  |   stack_ptr fork_top; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-13 15:00:05 +01:00
										 |  |  |   jv path; | 
					
						
							|  |  |  |   int subexp_nest; | 
					
						
							| 
									
										
										
										
											2013-05-05 23:12:10 +01:00
										 |  |  |   int debug_trace_enabled; | 
					
						
							| 
									
										
										
										
											2013-05-15 01:23:06 +01:00
										 |  |  |   int initial_execution; | 
					
						
							| 
									
										
										
										
											2014-07-09 00:55:20 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |   jv attrs; | 
					
						
							| 
									
										
										
											
												Add Streaming parser (--stream)
Streaming means that outputs are produced as soon as possible.  With the
`foreach` syntax one can write programs which reduce portions of the
streaming parse of a large input (reduce into proper JSON values, for
example), and discard the rest, processing incrementally.
This:
    $ jq -c --stream .
should produce the same output as this:
    $ jq -c '. as $dot | path(..) as $p | $dot | getpath($p) | [$p,.]'
The output of `jq --stream .` should be a sequence of`[[<path>],<leaf>]`
and `[[<path>]]` values.  The latter indicate that the array/object at
that path ended.
Scalars and empty arrays and objects are leaf values for this purpose.
For example, a truncated input produces a path as soon as possible, then
later the error:
    $ printf '[0,\n'|./jq -c --stream .
    [[0],0]
    parse error: Unfinished JSON term at EOF at line 3, column 0
    $
											
										 
											2014-12-22 23:06:27 -06:00
										 |  |  |   jq_input_cb input_cb; | 
					
						
							|  |  |  |   void *input_cb_data; | 
					
						
							| 
									
										
										
										
											2013-04-28 18:33:45 -05:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2012-08-16 01:00:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-14 22:08:18 +01:00
										 |  |  | struct closure { | 
					
						
							| 
									
										
										
										
											2014-06-30 19:40:56 -05:00
										 |  |  |   struct bytecode* bc;  // jq bytecode
 | 
					
						
							|  |  |  |   stack_ptr env;        // jq stack address of closed frame
 | 
					
						
							| 
									
										
										
										
											2013-06-14 22:08:18 +01:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-30 19:40:56 -05:00
										 |  |  | // locals for any function called: either a closure or a local variable
 | 
					
						
							| 
									
										
										
										
											2013-06-14 22:08:18 +01:00
										 |  |  | union frame_entry { | 
					
						
							|  |  |  |   struct closure closure; | 
					
						
							|  |  |  |   jv localvar; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-30 19:40:56 -05:00
										 |  |  | // jq function call frame
 | 
					
						
							| 
									
										
										
										
											2013-06-14 22:08:18 +01:00
										 |  |  | struct frame { | 
					
						
							| 
									
										
										
										
											2014-06-30 19:40:56 -05:00
										 |  |  |   struct bytecode* bc;      // jq bytecode for callee
 | 
					
						
							|  |  |  |   stack_ptr env;            // jq stack address of frame to return to
 | 
					
						
							|  |  |  |   stack_ptr retdata;        // jq stack address to unwind to on RET
 | 
					
						
							|  |  |  |   uint16_t* retaddr;        // jq bytecode return address
 | 
					
						
							|  |  |  |   union frame_entry entries[0]; // nclosures + nlocals
 | 
					
						
							| 
									
										
										
										
											2013-06-14 22:08:18 +01:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int frame_size(struct bytecode* bc) { | 
					
						
							|  |  |  |   return sizeof(struct frame) + sizeof(union frame_entry) * (bc->nclosures + bc->nlocals); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct frame* frame_current(struct jq_state* jq) { | 
					
						
							|  |  |  |   struct frame* fp = stack_block(&jq->stk, jq->curr_frame); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   stack_ptr next = *stack_block_next(&jq->stk, jq->curr_frame); | 
					
						
							|  |  |  |   if (next) { | 
					
						
							|  |  |  |     struct frame* fpnext = stack_block(&jq->stk, next); | 
					
						
							|  |  |  |     struct bytecode* bc = fpnext->bc; | 
					
						
							|  |  |  |     assert(fp->retaddr >= bc->code && fp->retaddr < bc->code + bc->codelen); | 
					
						
							|  |  |  |   } else { | 
					
						
							|  |  |  |     assert(fp->retaddr == 0); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return fp; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static stack_ptr frame_get_level(struct jq_state* jq, int level) { | 
					
						
							|  |  |  |   stack_ptr fr = jq->curr_frame; | 
					
						
							|  |  |  |   for (int i=0; i<level; i++) { | 
					
						
							|  |  |  |     struct frame* fp = stack_block(&jq->stk, fr); | 
					
						
							|  |  |  |     fr = fp->env; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return fr; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static jv* frame_local_var(struct jq_state* jq, int var, int level) { | 
					
						
							|  |  |  |   struct frame* fr = stack_block(&jq->stk, frame_get_level(jq, level)); | 
					
						
							|  |  |  |   assert(var >= 0); | 
					
						
							|  |  |  |   assert(var < fr->bc->nlocals); | 
					
						
							|  |  |  |   return &fr->entries[fr->bc->nclosures + var].localvar; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct closure make_closure(struct jq_state* jq, uint16_t* pc) { | 
					
						
							|  |  |  |   uint16_t level = *pc++; | 
					
						
							|  |  |  |   uint16_t idx = *pc++; | 
					
						
							|  |  |  |   stack_ptr fridx = frame_get_level(jq, level); | 
					
						
							|  |  |  |   struct frame* fr = stack_block(&jq->stk, fridx); | 
					
						
							|  |  |  |   if (idx & ARG_NEWCLOSURE) { | 
					
						
							| 
									
										
										
										
											2014-06-30 23:41:20 -05:00
										 |  |  |     // A new closure closing the frame identified by level, and with
 | 
					
						
							|  |  |  |     // the bytecode body of the idx'th subfunction of that frame
 | 
					
						
							| 
									
										
										
										
											2013-06-14 22:08:18 +01:00
										 |  |  |     int subfn_idx = idx & ~ARG_NEWCLOSURE; | 
					
						
							|  |  |  |     assert(subfn_idx < fr->bc->nsubfunctions); | 
					
						
							|  |  |  |     struct closure cl = {fr->bc->subfunctions[subfn_idx], | 
					
						
							|  |  |  |                          fridx}; | 
					
						
							|  |  |  |     return cl; | 
					
						
							|  |  |  |   } else { | 
					
						
							| 
									
										
										
										
											2014-06-30 23:41:20 -05:00
										 |  |  |     // A reference to a closure from the frame identified by level; copy
 | 
					
						
							|  |  |  |     // it as-is
 | 
					
						
							| 
									
										
										
										
											2013-06-14 22:08:18 +01:00
										 |  |  |     int closure = idx; | 
					
						
							|  |  |  |     assert(closure >= 0); | 
					
						
							|  |  |  |     assert(closure < fr->bc->nclosures); | 
					
						
							|  |  |  |     return fr->entries[closure].closure; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct frame* frame_push(struct jq_state* jq, struct closure callee,  | 
					
						
							|  |  |  |                                 uint16_t* argdef, int nargs) { | 
					
						
							|  |  |  |   stack_ptr new_frame_idx = stack_push_block(&jq->stk, jq->curr_frame, frame_size(callee.bc)); | 
					
						
							|  |  |  |   struct frame* new_frame = stack_block(&jq->stk, new_frame_idx); | 
					
						
							|  |  |  |   new_frame->bc = callee.bc; | 
					
						
							|  |  |  |   new_frame->env = callee.env; | 
					
						
							|  |  |  |   assert(nargs == new_frame->bc->nclosures); | 
					
						
							|  |  |  |   union frame_entry* entries = new_frame->entries; | 
					
						
							|  |  |  |   for (int i=0; i<nargs; i++) { | 
					
						
							|  |  |  |     entries->closure = make_closure(jq, argdef + i * 2); | 
					
						
							|  |  |  |     entries++; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   for (int i=0; i<callee.bc->nlocals; i++) { | 
					
						
							|  |  |  |     entries->localvar = jv_invalid(); | 
					
						
							|  |  |  |     entries++; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   jq->curr_frame = new_frame_idx; | 
					
						
							|  |  |  |   return new_frame; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void frame_pop(struct jq_state* jq) { | 
					
						
							|  |  |  |   assert(jq->curr_frame); | 
					
						
							|  |  |  |   struct frame* fp = frame_current(jq); | 
					
						
							|  |  |  |   if (stack_pop_will_free(&jq->stk, jq->curr_frame)) { | 
					
						
							|  |  |  |     int nlocals = fp->bc->nlocals; | 
					
						
							|  |  |  |     for (int i=0; i<nlocals; i++) { | 
					
						
							|  |  |  |       jv_free(*frame_local_var(jq, i, 0)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   jq->curr_frame = stack_pop_block(&jq->stk, jq->curr_frame, frame_size(fp->bc)); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2012-08-16 01:00:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-13 15:00:05 +01:00
										 |  |  | void stack_push(jq_state *jq, jv val) { | 
					
						
							|  |  |  |   assert(jv_is_valid(val)); | 
					
						
							| 
									
										
										
										
											2013-06-13 21:50:32 +01:00
										 |  |  |   jq->stk_top = stack_push_block(&jq->stk, jq->stk_top, sizeof(jv)); | 
					
						
							|  |  |  |   jv* sval = stack_block(&jq->stk, jq->stk_top); | 
					
						
							|  |  |  |   *sval = val; | 
					
						
							| 
									
										
										
										
											2012-08-16 01:00:30 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-13 15:00:05 +01:00
										 |  |  | jv stack_pop(jq_state *jq) { | 
					
						
							| 
									
										
										
										
											2013-06-13 21:50:32 +01:00
										 |  |  |   jv* sval = stack_block(&jq->stk, jq->stk_top); | 
					
						
							|  |  |  |   jv val = *sval; | 
					
						
							|  |  |  |   if (!stack_pop_will_free(&jq->stk, jq->stk_top)) { | 
					
						
							| 
									
										
										
										
											2013-05-13 15:00:05 +01:00
										 |  |  |     val = jv_copy(val); | 
					
						
							| 
									
										
										
										
											2012-09-02 17:20:13 +01:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2013-06-13 21:50:32 +01:00
										 |  |  |   jq->stk_top = stack_pop_block(&jq->stk, jq->stk_top, sizeof(jv)); | 
					
						
							| 
									
										
										
										
											2013-05-13 15:00:05 +01:00
										 |  |  |   assert(jv_is_valid(val)); | 
					
						
							|  |  |  |   return val; | 
					
						
							| 
									
										
										
										
											2012-08-16 01:00:30 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-29 15:24:43 -06:00
										 |  |  | // Like stack_pop(), but assert !stack_pop_will_free() and replace with
 | 
					
						
							|  |  |  | // jv_null() on the stack.
 | 
					
						
							|  |  |  | jv stack_popn(jq_state *jq) { | 
					
						
							|  |  |  |   jv* sval = stack_block(&jq->stk, jq->stk_top); | 
					
						
							|  |  |  |   jv val = *sval; | 
					
						
							|  |  |  |   if (!stack_pop_will_free(&jq->stk, jq->stk_top)) { | 
					
						
							|  |  |  |     *sval = jv_null(); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   jq->stk_top = stack_pop_block(&jq->stk, jq->stk_top, sizeof(jv)); | 
					
						
							|  |  |  |   assert(jv_is_valid(val)); | 
					
						
							|  |  |  |   return val; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-08-16 01:00:30 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | struct forkpoint { | 
					
						
							| 
									
										
										
										
											2013-06-13 21:50:32 +01:00
										 |  |  |   stack_ptr saved_data_stack; | 
					
						
							|  |  |  |   stack_ptr saved_curr_frame; | 
					
						
							| 
									
										
										
										
											2013-05-13 15:00:05 +01:00
										 |  |  |   int path_len, subexp_nest; | 
					
						
							| 
									
										
										
										
											2013-05-13 15:37:57 +01:00
										 |  |  |   uint16_t* return_address; | 
					
						
							| 
									
										
										
										
											2012-08-16 01:00:30 +01:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-13 21:50:32 +01:00
										 |  |  | struct stack_pos { | 
					
						
							|  |  |  |   stack_ptr saved_data_stack, saved_curr_frame; | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2012-08-16 01:00:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-13 21:50:32 +01:00
										 |  |  | struct stack_pos stack_get_pos(jq_state* jq) { | 
					
						
							|  |  |  |   struct stack_pos sp = {jq->stk_top, jq->curr_frame}; | 
					
						
							|  |  |  |   return sp; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2012-08-16 01:00:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-13 21:50:32 +01:00
										 |  |  | void stack_save(jq_state *jq, uint16_t* retaddr, struct stack_pos sp){ | 
					
						
							|  |  |  |   jq->fork_top = stack_push_block(&jq->stk, jq->fork_top, sizeof(struct forkpoint)); | 
					
						
							|  |  |  |   struct forkpoint* fork = stack_block(&jq->stk, jq->fork_top); | 
					
						
							|  |  |  |   fork->saved_data_stack = jq->stk_top; | 
					
						
							|  |  |  |   fork->saved_curr_frame = jq->curr_frame; | 
					
						
							| 
									
										
										
										
											2013-05-13 15:00:05 +01:00
										 |  |  |   fork->path_len =  | 
					
						
							|  |  |  |     jv_get_kind(jq->path) == JV_KIND_ARRAY ? jv_array_length(jv_copy(jq->path)) : 0; | 
					
						
							|  |  |  |   fork->subexp_nest = jq->subexp_nest; | 
					
						
							| 
									
										
										
										
											2013-05-13 15:37:57 +01:00
										 |  |  |   fork->return_address = retaddr; | 
					
						
							| 
									
										
										
										
											2013-06-13 21:50:32 +01:00
										 |  |  |   jq->stk_top = sp.saved_data_stack; | 
					
						
							|  |  |  |   jq->curr_frame = sp.saved_curr_frame; | 
					
						
							| 
									
										
										
										
											2012-08-16 01:00:30 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-13 15:00:05 +01:00
										 |  |  | void path_append(jq_state* jq, jv component) { | 
					
						
							|  |  |  |   if (jq->subexp_nest == 0 && jv_get_kind(jq->path) == JV_KIND_ARRAY) { | 
					
						
							|  |  |  |     int n1 = jv_array_length(jv_copy(jq->path)); | 
					
						
							|  |  |  |     jq->path = jv_array_append(jq->path, component); | 
					
						
							|  |  |  |     int n2 = jv_array_length(jv_copy(jq->path)); | 
					
						
							|  |  |  |     assert(n2 == n1 + 1); | 
					
						
							|  |  |  |   } else { | 
					
						
							|  |  |  |     jv_free(component); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-13 15:37:57 +01:00
										 |  |  | uint16_t* stack_restore(jq_state *jq){ | 
					
						
							| 
									
										
										
										
											2013-06-14 22:08:18 +01:00
										 |  |  |   while (!stack_pop_will_free(&jq->stk, jq->fork_top)) { | 
					
						
							| 
									
										
										
										
											2013-06-13 21:50:32 +01:00
										 |  |  |     if (stack_pop_will_free(&jq->stk, jq->stk_top)) { | 
					
						
							| 
									
										
										
										
											2013-06-10 01:17:58 +01:00
										 |  |  |       jv_free(stack_pop(jq)); | 
					
						
							| 
									
										
										
										
											2013-06-13 21:50:32 +01:00
										 |  |  |     } else if (stack_pop_will_free(&jq->stk, jq->curr_frame)) { | 
					
						
							| 
									
										
										
										
											2013-06-14 22:08:18 +01:00
										 |  |  |       frame_pop(jq); | 
					
						
							| 
									
										
										
										
											2013-06-10 01:17:58 +01:00
										 |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2013-06-13 21:50:32 +01:00
										 |  |  |       assert(0); | 
					
						
							| 
									
										
										
										
											2013-06-10 01:17:58 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2012-09-02 21:45:27 +01:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2013-05-13 15:00:05 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-13 21:50:32 +01:00
										 |  |  |   if (jq->fork_top == 0) { | 
					
						
							| 
									
										
										
										
											2012-09-02 22:08:36 +01:00
										 |  |  |     return 0; | 
					
						
							| 
									
										
										
										
											2013-05-13 15:00:05 +01:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-13 21:50:32 +01:00
										 |  |  |   struct forkpoint* fork = stack_block(&jq->stk, jq->fork_top); | 
					
						
							| 
									
										
										
										
											2013-05-13 15:37:57 +01:00
										 |  |  |   uint16_t* retaddr = fork->return_address; | 
					
						
							| 
									
										
										
										
											2013-06-13 21:50:32 +01:00
										 |  |  |   jq->stk_top = fork->saved_data_stack; | 
					
						
							| 
									
										
										
										
											2013-06-10 01:17:58 +01:00
										 |  |  |   jq->curr_frame = fork->saved_curr_frame; | 
					
						
							| 
									
										
										
										
											2013-05-13 15:00:05 +01:00
										 |  |  |   int path_len = fork->path_len; | 
					
						
							|  |  |  |   if (jv_get_kind(jq->path) == JV_KIND_ARRAY) { | 
					
						
							|  |  |  |     assert(path_len >= 0); | 
					
						
							|  |  |  |     jq->path = jv_array_slice(jq->path, 0, path_len); | 
					
						
							| 
									
										
										
										
											2012-09-02 22:08:36 +01:00
										 |  |  |   } else { | 
					
						
							| 
									
										
										
										
											2013-05-13 15:00:05 +01:00
										 |  |  |     assert(path_len == 0); | 
					
						
							| 
									
										
										
										
											2012-09-02 22:08:36 +01:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2013-05-13 15:00:05 +01:00
										 |  |  |   jq->subexp_nest = fork->subexp_nest; | 
					
						
							| 
									
										
										
										
											2013-06-13 21:50:32 +01:00
										 |  |  |   jq->fork_top = stack_pop_block(&jq->stk, jq->fork_top, sizeof(struct forkpoint)); | 
					
						
							| 
									
										
										
										
											2013-05-13 15:37:57 +01:00
										 |  |  |   return retaddr; | 
					
						
							| 
									
										
										
										
											2012-08-16 01:00:30 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-06 17:26:15 -05:00
										 |  |  | static void jq_reset(jq_state *jq) { | 
					
						
							|  |  |  |   while (stack_restore(jq)) {} | 
					
						
							| 
									
										
										
										
											2013-06-18 00:06:00 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |   assert(jq->stk_top == 0); | 
					
						
							|  |  |  |   assert(jq->fork_top == 0); | 
					
						
							|  |  |  |   assert(jq->curr_frame == 0); | 
					
						
							|  |  |  |   stack_reset(&jq->stk); | 
					
						
							| 
									
										
										
										
											2014-07-05 20:54:42 -05:00
										 |  |  |   jv_free(jq->error); | 
					
						
							|  |  |  |   jq->error = jv_null(); | 
					
						
							| 
									
										
										
										
											2013-06-06 17:26:15 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (jv_get_kind(jq->path) != JV_KIND_INVALID) | 
					
						
							|  |  |  |     jv_free(jq->path); | 
					
						
							|  |  |  |   jq->path = jv_null(); | 
					
						
							| 
									
										
										
										
											2013-06-18 00:17:20 +01:00
										 |  |  |   jq->subexp_nest = 0; | 
					
						
							| 
									
										
										
										
											2013-06-06 17:26:15 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-08-11 17:25:09 -05:00
										 |  |  | void jq_report_error(jq_state *jq, jv value) { | 
					
						
							|  |  |  |   assert(jq->err_cb); | 
					
						
							|  |  |  |   // callback must jv_free() its jv argument
 | 
					
						
							|  |  |  |   jq->err_cb(jq->err_cb_data, value); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void set_error(jq_state *jq, jv value) { | 
					
						
							|  |  |  |   // Record so try/catch can find it.
 | 
					
						
							| 
									
										
										
										
											2014-07-05 20:54:42 -05:00
										 |  |  |   jv_free(jq->error); | 
					
						
							|  |  |  |   jq->error = value; | 
					
						
							| 
									
										
										
										
											2012-09-11 10:37:44 +01:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2014-08-11 17:25:09 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-08-16 01:00:30 +01:00
										 |  |  | #define ON_BACKTRACK(op) ((op)+NUM_OPCODES)
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-28 18:33:45 -05:00
										 |  |  | jv jq_next(jq_state *jq) { | 
					
						
							| 
									
										
										
										
											2012-09-02 17:20:13 +01:00
										 |  |  |   jv cfunc_input[MAX_CFUNCTION_ARGS]; | 
					
						
							| 
									
										
										
										
											2012-08-21 12:35:36 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-06 17:26:15 -05:00
										 |  |  |   jv_nomem_handler(jq->nomem_handler, jq->nomem_handler_data); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-13 15:37:57 +01:00
										 |  |  |   uint16_t* pc = stack_restore(jq); | 
					
						
							|  |  |  |   assert(pc); | 
					
						
							| 
									
										
										
										
											2012-08-21 12:35:36 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-07-10 19:20:58 -05:00
										 |  |  |   int raising; | 
					
						
							| 
									
										
										
										
											2013-05-15 01:23:06 +01:00
										 |  |  |   int backtracking = !jq->initial_execution; | 
					
						
							|  |  |  |   jq->initial_execution = 0; | 
					
						
							| 
									
										
										
										
											2014-07-05 20:54:42 -05:00
										 |  |  |   assert(jv_get_kind(jq->error) == JV_KIND_NULL); | 
					
						
							| 
									
										
										
										
											2012-08-16 01:00:30 +01:00
										 |  |  |   while (1) { | 
					
						
							| 
									
										
										
										
											2012-09-10 10:23:15 +01:00
										 |  |  |     uint16_t opcode = *pc; | 
					
						
							| 
									
										
										
										
											2014-07-10 19:20:58 -05:00
										 |  |  |     raising = 0; | 
					
						
							| 
									
										
										
										
											2012-08-16 01:00:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-05 23:12:10 +01:00
										 |  |  |     if (jq->debug_trace_enabled) { | 
					
						
							| 
									
										
										
										
											2013-06-14 22:08:18 +01:00
										 |  |  |       dump_operation(frame_current(jq)->bc, pc); | 
					
						
							| 
									
										
										
										
											2013-05-05 23:12:10 +01:00
										 |  |  |       printf("\t"); | 
					
						
							|  |  |  |       const struct opcode_description* opdesc = opcode_describe(opcode); | 
					
						
							| 
									
										
										
										
											2013-06-13 21:50:32 +01:00
										 |  |  |       stack_ptr param = 0; | 
					
						
							| 
									
										
										
										
											2013-06-10 01:17:58 +01:00
										 |  |  |       if (!backtracking) { | 
					
						
							|  |  |  |         int stack_in = opdesc->stack_in; | 
					
						
							|  |  |  |         if (stack_in == -1) stack_in = pc[1]; | 
					
						
							|  |  |  |         for (int i=0; i<stack_in; i++) { | 
					
						
							|  |  |  |           if (i == 0) { | 
					
						
							| 
									
										
										
										
											2013-06-13 21:50:32 +01:00
										 |  |  |             param = jq->stk_top; | 
					
						
							| 
									
										
										
										
											2013-06-10 01:17:58 +01:00
										 |  |  |           } else { | 
					
						
							|  |  |  |             printf(" | "); | 
					
						
							| 
									
										
										
										
											2013-06-13 21:50:32 +01:00
										 |  |  |             param = *stack_block_next(&jq->stk, param); | 
					
						
							| 
									
										
										
										
											2013-06-10 01:17:58 +01:00
										 |  |  |           } | 
					
						
							|  |  |  |           if (!param) break; | 
					
						
							| 
									
										
										
										
											2014-11-28 19:35:29 -06:00
										 |  |  |           jv_dump(jv_copy(*(jv*)stack_block(&jq->stk, param)), JV_PRINT_REFCOUNT); | 
					
						
							| 
									
										
										
										
											2013-06-10 01:17:58 +01:00
										 |  |  |           //printf("<%d>", jv_get_refcnt(param->val));
 | 
					
						
							|  |  |  |           //printf(" -- ");
 | 
					
						
							|  |  |  |           //jv_dump(jv_copy(jq->path), 0);
 | 
					
						
							| 
									
										
										
										
											2013-05-05 23:12:10 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2013-06-10 01:17:58 +01:00
										 |  |  |       } else { | 
					
						
							|  |  |  |         printf("\t<backtracking>"); | 
					
						
							| 
									
										
										
										
											2012-08-21 12:35:36 +01:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2012-08-16 01:00:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-05 23:12:10 +01:00
										 |  |  |       printf("\n"); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2013-06-13 21:50:32 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-08-16 01:00:30 +01:00
										 |  |  |     if (backtracking) { | 
					
						
							|  |  |  |       opcode = ON_BACKTRACK(opcode); | 
					
						
							|  |  |  |       backtracking = 0; | 
					
						
							| 
									
										
										
										
											2014-07-10 19:20:58 -05:00
										 |  |  |       raising = !jv_is_valid(jq->error); | 
					
						
							| 
									
										
										
										
											2012-08-16 01:00:30 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2012-09-10 10:23:15 +01:00
										 |  |  |     pc++; | 
					
						
							| 
									
										
										
										
											2012-08-16 01:00:30 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     switch (opcode) { | 
					
						
							|  |  |  |     default: assert(0 && "invalid instruction"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-07-07 16:03:32 -05:00
										 |  |  |     case TOP: break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-08-16 01:00:30 +01:00
										 |  |  |     case LOADK: { | 
					
						
							| 
									
										
										
										
											2013-06-14 22:08:18 +01:00
										 |  |  |       jv v = jv_array_get(jv_copy(frame_current(jq)->bc->constants), *pc++); | 
					
						
							| 
									
										
										
										
											2012-09-03 17:26:52 +01:00
										 |  |  |       assert(jv_is_valid(v)); | 
					
						
							| 
									
										
										
										
											2013-05-13 15:00:05 +01:00
										 |  |  |       jv_free(stack_pop(jq)); | 
					
						
							|  |  |  |       stack_push(jq, v); | 
					
						
							| 
									
										
										
										
											2012-08-16 01:00:30 +01:00
										 |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case DUP: { | 
					
						
							| 
									
										
										
										
											2013-05-13 15:00:05 +01:00
										 |  |  |       jv v = stack_pop(jq); | 
					
						
							|  |  |  |       stack_push(jq, jv_copy(v)); | 
					
						
							| 
									
										
										
										
											2013-04-28 18:33:45 -05:00
										 |  |  |       stack_push(jq, v); | 
					
						
							| 
									
										
										
										
											2012-08-16 01:00:30 +01:00
										 |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-29 15:24:43 -06:00
										 |  |  |     case DUPN: { | 
					
						
							|  |  |  |       jv v = stack_popn(jq); | 
					
						
							|  |  |  |       stack_push(jq, jv_copy(v)); | 
					
						
							|  |  |  |       stack_push(jq, v); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-03 20:31:40 +00:00
										 |  |  |     case DUP2: { | 
					
						
							| 
									
										
										
										
											2013-05-13 15:00:05 +01:00
										 |  |  |       jv keep = stack_pop(jq); | 
					
						
							|  |  |  |       jv v = stack_pop(jq); | 
					
						
							|  |  |  |       stack_push(jq, jv_copy(v)); | 
					
						
							| 
									
										
										
										
											2013-04-28 18:33:45 -05:00
										 |  |  |       stack_push(jq, keep); | 
					
						
							| 
									
										
										
										
											2013-05-13 15:00:05 +01:00
										 |  |  |       stack_push(jq, v); | 
					
						
							| 
									
										
										
										
											2012-12-03 20:31:40 +00:00
										 |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-13 15:00:05 +01:00
										 |  |  |     case SUBEXP_BEGIN: { | 
					
						
							|  |  |  |       jv v = stack_pop(jq); | 
					
						
							|  |  |  |       stack_push(jq, jv_copy(v)); | 
					
						
							|  |  |  |       stack_push(jq, v); | 
					
						
							|  |  |  |       jq->subexp_nest++; | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case SUBEXP_END: { | 
					
						
							|  |  |  |       assert(jq->subexp_nest > 0); | 
					
						
							|  |  |  |       jq->subexp_nest--; | 
					
						
							|  |  |  |       jv a = stack_pop(jq); | 
					
						
							|  |  |  |       jv b = stack_pop(jq); | 
					
						
							| 
									
										
										
										
											2013-04-28 18:33:45 -05:00
										 |  |  |       stack_push(jq, a); | 
					
						
							|  |  |  |       stack_push(jq, b); | 
					
						
							| 
									
										
										
										
											2012-08-16 01:00:30 +01:00
										 |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |        | 
					
						
							|  |  |  |     case POP: { | 
					
						
							| 
									
										
										
										
											2013-05-13 15:00:05 +01:00
										 |  |  |       jv_free(stack_pop(jq)); | 
					
						
							| 
									
										
										
										
											2012-08-16 01:00:30 +01:00
										 |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case APPEND: { | 
					
						
							| 
									
										
										
										
											2013-05-13 15:00:05 +01:00
										 |  |  |       jv v = stack_pop(jq); | 
					
						
							| 
									
										
										
										
											2013-01-03 12:51:33 +00:00
										 |  |  |       uint16_t level = *pc++; | 
					
						
							|  |  |  |       uint16_t vidx = *pc++; | 
					
						
							| 
									
										
										
										
											2013-06-14 22:08:18 +01:00
										 |  |  |       jv* var = frame_local_var(jq, vidx, level); | 
					
						
							| 
									
										
										
										
											2013-01-03 12:51:33 +00:00
										 |  |  |       assert(jv_get_kind(*var) == JV_KIND_ARRAY); | 
					
						
							|  |  |  |       *var = jv_array_append(*var, v); | 
					
						
							| 
									
										
										
										
											2012-08-16 01:00:30 +01:00
										 |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case INSERT: { | 
					
						
							| 
									
										
										
										
											2013-05-13 15:00:05 +01:00
										 |  |  |       jv stktop = stack_pop(jq); | 
					
						
							|  |  |  |       jv v = stack_pop(jq); | 
					
						
							|  |  |  |       jv k = stack_pop(jq); | 
					
						
							|  |  |  |       jv objv = stack_pop(jq); | 
					
						
							|  |  |  |       assert(jv_get_kind(objv) == JV_KIND_OBJECT); | 
					
						
							| 
									
										
										
										
											2013-05-13 15:42:57 +01:00
										 |  |  |       if (jv_get_kind(k) == JV_KIND_STRING) { | 
					
						
							|  |  |  |         stack_push(jq, jv_object_set(objv, k, v)); | 
					
						
							|  |  |  |         stack_push(jq, stktop); | 
					
						
							|  |  |  |       } else { | 
					
						
							| 
									
										
										
										
											2014-08-11 17:25:09 -05:00
										 |  |  |         set_error(jq, jv_invalid_with_msg(jv_string_fmt("Cannot use %s as object key", | 
					
						
							|  |  |  |                                                         jv_kind_name(jv_get_kind(k))))); | 
					
						
							| 
									
										
										
										
											2013-05-13 15:42:57 +01:00
										 |  |  |         jv_free(stktop); | 
					
						
							|  |  |  |         jv_free(v); | 
					
						
							|  |  |  |         jv_free(k); | 
					
						
							|  |  |  |         jv_free(objv); | 
					
						
							|  |  |  |         goto do_backtrack; | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2012-08-16 01:00:30 +01:00
										 |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-14 16:09:10 +01:00
										 |  |  |     case ON_BACKTRACK(RANGE): | 
					
						
							|  |  |  |     case RANGE: { | 
					
						
							|  |  |  |       uint16_t level = *pc++; | 
					
						
							|  |  |  |       uint16_t v = *pc++; | 
					
						
							| 
									
										
										
										
											2013-06-14 22:08:18 +01:00
										 |  |  |       jv* var = frame_local_var(jq, v, level); | 
					
						
							| 
									
										
										
										
											2013-05-14 16:09:10 +01:00
										 |  |  |       jv max = stack_pop(jq); | 
					
						
							| 
									
										
										
										
											2014-07-10 19:20:58 -05:00
										 |  |  |       if (raising) goto do_backtrack; | 
					
						
							| 
									
										
										
										
											2013-05-14 16:09:10 +01:00
										 |  |  |       if (jv_get_kind(*var) != JV_KIND_NUMBER || | 
					
						
							|  |  |  |           jv_get_kind(max) != JV_KIND_NUMBER) { | 
					
						
							| 
									
										
										
										
											2014-08-11 17:25:09 -05:00
										 |  |  |         set_error(jq, jv_invalid_with_msg(jv_string_fmt("Range bounds must be numeric"))); | 
					
						
							| 
									
										
										
										
											2013-05-14 16:09:10 +01:00
										 |  |  |         jv_free(max); | 
					
						
							|  |  |  |         goto do_backtrack; | 
					
						
							|  |  |  |       } else if (jv_number_value(jv_copy(*var)) >= jv_number_value(jv_copy(max))) { | 
					
						
							|  |  |  |         /* finished iterating */ | 
					
						
							|  |  |  |         goto do_backtrack; | 
					
						
							|  |  |  |       } else { | 
					
						
							|  |  |  |         jv curr = jv_copy(*var); | 
					
						
							|  |  |  |         *var = jv_number(jv_number_value(*var) + 1); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-13 21:50:32 +01:00
										 |  |  |         struct stack_pos spos = stack_get_pos(jq); | 
					
						
							| 
									
										
										
										
											2013-05-14 16:09:10 +01:00
										 |  |  |         stack_push(jq, jv_copy(max)); | 
					
						
							| 
									
										
										
										
											2013-06-13 21:50:32 +01:00
										 |  |  |         stack_save(jq, pc - 3, spos); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-14 16:09:10 +01:00
										 |  |  |         stack_push(jq, curr); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-09-02 17:20:13 +01:00
										 |  |  |       // FIXME: loadv/storev may do too much copying/freeing
 | 
					
						
							| 
									
										
										
										
											2012-08-16 01:00:30 +01:00
										 |  |  |     case LOADV: { | 
					
						
							| 
									
										
										
										
											2012-08-21 18:24:38 +01:00
										 |  |  |       uint16_t level = *pc++; | 
					
						
							| 
									
										
										
										
											2012-08-16 01:00:30 +01:00
										 |  |  |       uint16_t v = *pc++; | 
					
						
							| 
									
										
										
										
											2013-06-14 22:08:18 +01:00
										 |  |  |       jv* var = frame_local_var(jq, v, level); | 
					
						
							| 
									
										
										
										
											2013-05-05 23:12:10 +01:00
										 |  |  |       if (jq->debug_trace_enabled) { | 
					
						
							|  |  |  |         printf("V%d = ", v); | 
					
						
							|  |  |  |         jv_dump(jv_copy(*var), 0); | 
					
						
							| 
									
										
										
										
											2014-11-28 20:52:08 -06:00
										 |  |  |         printf(" (%d)\n", jv_get_refcnt(*var)); | 
					
						
							| 
									
										
										
										
											2013-05-05 23:12:10 +01:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2013-05-13 15:00:05 +01:00
										 |  |  |       jv_free(stack_pop(jq)); | 
					
						
							|  |  |  |       stack_push(jq, jv_copy(*var)); | 
					
						
							| 
									
										
										
										
											2012-08-16 01:00:30 +01:00
										 |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-16 14:32:18 +01:00
										 |  |  |       // Does a load but replaces the variable with null
 | 
					
						
							|  |  |  |     case LOADVN: { | 
					
						
							|  |  |  |       uint16_t level = *pc++; | 
					
						
							|  |  |  |       uint16_t v = *pc++; | 
					
						
							| 
									
										
										
										
											2013-06-14 22:08:18 +01:00
										 |  |  |       jv* var = frame_local_var(jq, v, level); | 
					
						
							| 
									
										
										
										
											2013-05-16 14:32:18 +01:00
										 |  |  |       if (jq->debug_trace_enabled) { | 
					
						
							|  |  |  |         printf("V%d = ", v); | 
					
						
							|  |  |  |         jv_dump(jv_copy(*var), 0); | 
					
						
							| 
									
										
										
										
											2014-11-28 20:52:08 -06:00
										 |  |  |         printf(" (%d)\n", jv_get_refcnt(*var)); | 
					
						
							| 
									
										
										
										
											2013-05-16 14:32:18 +01:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2014-11-29 15:24:43 -06:00
										 |  |  |       jv_free(stack_popn(jq)); | 
					
						
							| 
									
										
										
										
											2013-05-16 14:32:18 +01:00
										 |  |  |       stack_push(jq, *var); | 
					
						
							|  |  |  |       *var = jv_null(); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-08-16 01:00:30 +01:00
										 |  |  |     case STOREV: { | 
					
						
							| 
									
										
										
										
											2012-08-21 18:24:38 +01:00
										 |  |  |       uint16_t level = *pc++; | 
					
						
							| 
									
										
										
										
											2012-08-16 01:00:30 +01:00
										 |  |  |       uint16_t v = *pc++; | 
					
						
							| 
									
										
										
										
											2013-06-14 22:08:18 +01:00
										 |  |  |       jv* var = frame_local_var(jq, v, level); | 
					
						
							| 
									
										
										
										
											2013-05-13 15:00:05 +01:00
										 |  |  |       jv val = stack_pop(jq); | 
					
						
							| 
									
										
										
										
											2013-05-05 23:12:10 +01:00
										 |  |  |       if (jq->debug_trace_enabled) { | 
					
						
							|  |  |  |         printf("V%d = ", v); | 
					
						
							| 
									
										
										
										
											2013-05-13 15:00:05 +01:00
										 |  |  |         jv_dump(jv_copy(val), 0); | 
					
						
							| 
									
										
										
										
											2014-11-28 20:52:08 -06:00
										 |  |  |         printf(" (%d)\n", jv_get_refcnt(val)); | 
					
						
							| 
									
										
										
										
											2013-05-05 23:12:10 +01:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2012-09-02 17:20:13 +01:00
										 |  |  |       jv_free(*var); | 
					
						
							| 
									
										
										
										
											2013-05-13 15:00:05 +01:00
										 |  |  |       *var = val; | 
					
						
							| 
									
										
										
										
											2012-08-16 01:00:30 +01:00
										 |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-13 15:00:05 +01:00
										 |  |  |     case PATH_BEGIN: { | 
					
						
							|  |  |  |       jv v = stack_pop(jq); | 
					
						
							|  |  |  |       stack_push(jq, jq->path); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-13 21:50:32 +01:00
										 |  |  |       stack_save(jq, pc - 1, stack_get_pos(jq)); | 
					
						
							| 
									
										
										
										
											2013-05-13 15:00:05 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |       stack_push(jq, jv_number(jq->subexp_nest)); | 
					
						
							|  |  |  |       stack_push(jq, v); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       jq->path = jv_array(); | 
					
						
							|  |  |  |       jq->subexp_nest = 0; | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case PATH_END: { | 
					
						
							|  |  |  |       jv v = stack_pop(jq); | 
					
						
							|  |  |  |       jv_free(v); // discard value, only keep path
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       int old_subexp_nest = (int)jv_number_value(stack_pop(jq)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       jv path = jq->path; | 
					
						
							|  |  |  |       jq->path = stack_pop(jq); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-13 21:50:32 +01:00
										 |  |  |       struct stack_pos spos = stack_get_pos(jq); | 
					
						
							| 
									
										
										
										
											2013-05-13 15:00:05 +01:00
										 |  |  |       stack_push(jq, jv_copy(path)); | 
					
						
							| 
									
										
										
										
											2013-06-13 21:50:32 +01:00
										 |  |  |       stack_save(jq, pc - 1, spos); | 
					
						
							| 
									
										
										
										
											2013-05-13 15:00:05 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |       stack_push(jq, path); | 
					
						
							|  |  |  |       jq->subexp_nest = old_subexp_nest; | 
					
						
							| 
									
										
										
										
											2012-12-28 16:08:29 +00:00
										 |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-13 15:00:05 +01:00
										 |  |  |     case ON_BACKTRACK(PATH_BEGIN): | 
					
						
							|  |  |  |     case ON_BACKTRACK(PATH_END): { | 
					
						
							|  |  |  |       jv_free(jq->path); | 
					
						
							|  |  |  |       jq->path = stack_pop(jq); | 
					
						
							|  |  |  |       goto do_backtrack; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-17 12:28:26 -06:00
										 |  |  |     case INDEX: | 
					
						
							|  |  |  |     case INDEX_OPT: { | 
					
						
							| 
									
										
										
										
											2013-05-13 15:00:05 +01:00
										 |  |  |       jv t = stack_pop(jq); | 
					
						
							|  |  |  |       jv k = stack_pop(jq); | 
					
						
							|  |  |  |       path_append(jq, jv_copy(k)); | 
					
						
							|  |  |  |       jv v = jv_get(t, k); | 
					
						
							| 
									
										
										
										
											2012-09-11 10:37:44 +01:00
										 |  |  |       if (jv_is_valid(v)) { | 
					
						
							| 
									
										
										
										
											2013-05-13 15:00:05 +01:00
										 |  |  |         stack_push(jq, v); | 
					
						
							| 
									
										
										
										
											2012-08-16 01:00:30 +01:00
										 |  |  |       } else { | 
					
						
							| 
									
										
										
										
											2014-02-17 12:28:26 -06:00
										 |  |  |         if (opcode == INDEX) | 
					
						
							| 
									
										
										
										
											2014-08-11 17:25:09 -05:00
										 |  |  |           set_error(jq, v); | 
					
						
							| 
									
										
										
										
											2014-02-17 12:28:26 -06:00
										 |  |  |         else | 
					
						
							|  |  |  |           jv_free(v); | 
					
						
							| 
									
										
										
										
											2012-09-11 10:37:44 +01:00
										 |  |  |         goto do_backtrack; | 
					
						
							| 
									
										
										
										
											2012-08-16 01:00:30 +01:00
										 |  |  |       } | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case JUMP: { | 
					
						
							|  |  |  |       uint16_t offset = *pc++; | 
					
						
							|  |  |  |       pc += offset; | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-09-04 15:34:34 +01:00
										 |  |  |     case JUMP_F: { | 
					
						
							|  |  |  |       uint16_t offset = *pc++; | 
					
						
							| 
									
										
										
										
											2013-05-13 15:00:05 +01:00
										 |  |  |       jv t = stack_pop(jq); | 
					
						
							|  |  |  |       jv_kind kind = jv_get_kind(t); | 
					
						
							| 
									
										
										
										
											2012-09-04 15:34:34 +01:00
										 |  |  |       if (kind == JV_KIND_FALSE || kind == JV_KIND_NULL) { | 
					
						
							|  |  |  |         pc += offset; | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2013-04-28 18:33:45 -05:00
										 |  |  |       stack_push(jq, t); // FIXME do this better
 | 
					
						
							| 
									
										
										
										
											2012-09-04 15:34:34 +01:00
										 |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-08-16 01:00:30 +01:00
										 |  |  |     case EACH:  | 
					
						
							| 
									
										
										
										
											2014-02-17 12:28:26 -06:00
										 |  |  |     case EACH_OPT:  | 
					
						
							| 
									
										
										
										
											2013-05-13 15:00:05 +01:00
										 |  |  |       stack_push(jq, jv_number(-1)); | 
					
						
							| 
									
										
										
										
											2012-08-16 01:00:30 +01:00
										 |  |  |       // fallthrough
 | 
					
						
							| 
									
										
										
										
											2014-02-17 12:28:26 -06:00
										 |  |  |     case ON_BACKTRACK(EACH): | 
					
						
							|  |  |  |     case ON_BACKTRACK(EACH_OPT): { | 
					
						
							| 
									
										
										
										
											2013-05-13 15:00:05 +01:00
										 |  |  |       int idx = jv_number_value(stack_pop(jq)); | 
					
						
							|  |  |  |       jv container = stack_pop(jq); | 
					
						
							| 
									
										
										
										
											2012-09-11 16:10:55 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-17 22:49:08 +01:00
										 |  |  |       int keep_going, is_last = 0; | 
					
						
							| 
									
										
										
										
											2012-09-11 16:10:55 +01:00
										 |  |  |       jv key, value; | 
					
						
							| 
									
										
										
										
											2013-05-13 15:00:05 +01:00
										 |  |  |       if (jv_get_kind(container) == JV_KIND_ARRAY) { | 
					
						
							| 
									
										
										
										
											2014-02-17 12:28:26 -06:00
										 |  |  |         if (opcode == EACH || opcode == EACH_OPT) idx = 0; | 
					
						
							| 
									
										
										
										
											2012-09-11 16:10:55 +01:00
										 |  |  |         else idx = idx + 1; | 
					
						
							| 
									
										
										
										
											2013-05-17 22:49:08 +01:00
										 |  |  |         int len = jv_array_length(jv_copy(container)); | 
					
						
							|  |  |  |         keep_going = idx < len; | 
					
						
							|  |  |  |         is_last = idx == len - 1; | 
					
						
							| 
									
										
										
										
											2012-09-11 16:10:55 +01:00
										 |  |  |         if (keep_going) { | 
					
						
							|  |  |  |           key = jv_number(idx); | 
					
						
							| 
									
										
										
										
											2013-05-13 15:00:05 +01:00
										 |  |  |           value = jv_array_get(jv_copy(container), idx); | 
					
						
							| 
									
										
										
										
											2012-09-11 16:10:55 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2013-05-13 15:00:05 +01:00
										 |  |  |       } else if (jv_get_kind(container) == JV_KIND_OBJECT) { | 
					
						
							| 
									
										
										
										
											2014-02-17 12:28:26 -06:00
										 |  |  |         if (opcode == EACH || opcode == EACH_OPT) idx = jv_object_iter(container); | 
					
						
							| 
									
										
										
										
											2013-05-13 15:00:05 +01:00
										 |  |  |         else idx = jv_object_iter_next(container, idx); | 
					
						
							|  |  |  |         keep_going = jv_object_iter_valid(container, idx); | 
					
						
							| 
									
										
										
										
											2012-09-11 16:10:55 +01:00
										 |  |  |         if (keep_going) { | 
					
						
							| 
									
										
										
										
											2013-05-13 15:00:05 +01:00
										 |  |  |           key = jv_object_iter_key(container, idx); | 
					
						
							|  |  |  |           value = jv_object_iter_value(container, idx); | 
					
						
							| 
									
										
										
										
											2012-09-11 16:10:55 +01:00
										 |  |  |         } | 
					
						
							|  |  |  |       } else { | 
					
						
							| 
									
										
										
										
											2014-02-17 12:28:26 -06:00
										 |  |  |         assert(opcode == EACH || opcode == EACH_OPT); | 
					
						
							|  |  |  |         if (opcode == EACH) { | 
					
						
							| 
									
										
										
										
											2014-08-11 17:25:09 -05:00
										 |  |  |           set_error(jq, | 
					
						
							|  |  |  |                     jv_invalid_with_msg(jv_string_fmt("Cannot iterate over %s", | 
					
						
							|  |  |  |                                                       jv_kind_name(jv_get_kind(container))))); | 
					
						
							| 
									
										
										
										
											2014-02-17 12:28:26 -06:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2012-09-11 16:10:55 +01:00
										 |  |  |         keep_going = 0; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-07-10 19:20:58 -05:00
										 |  |  |       if (!keep_going || raising) { | 
					
						
							|  |  |  |         if (keep_going) | 
					
						
							|  |  |  |           jv_free(value); | 
					
						
							| 
									
										
										
										
											2013-05-13 15:00:05 +01:00
										 |  |  |         jv_free(container); | 
					
						
							| 
									
										
										
										
											2012-08-16 01:00:30 +01:00
										 |  |  |         goto do_backtrack; | 
					
						
							| 
									
										
										
										
											2013-05-17 22:49:08 +01:00
										 |  |  |       } else if (is_last) { | 
					
						
							|  |  |  |         // we don't need to make a backtrack point
 | 
					
						
							|  |  |  |         jv_free(container); | 
					
						
							|  |  |  |         path_append(jq, key); | 
					
						
							|  |  |  |         stack_push(jq, value); | 
					
						
							| 
									
										
										
										
											2012-08-16 01:00:30 +01:00
										 |  |  |       } else { | 
					
						
							| 
									
										
										
										
											2013-06-13 21:50:32 +01:00
										 |  |  |         struct stack_pos spos = stack_get_pos(jq); | 
					
						
							| 
									
										
										
										
											2013-04-28 18:33:45 -05:00
										 |  |  |         stack_push(jq, container); | 
					
						
							| 
									
										
										
										
											2013-05-13 15:00:05 +01:00
										 |  |  |         stack_push(jq, jv_number(idx)); | 
					
						
							| 
									
										
										
										
											2013-06-13 21:50:32 +01:00
										 |  |  |         stack_save(jq, pc - 1, spos); | 
					
						
							| 
									
										
										
										
											2013-05-13 15:00:05 +01:00
										 |  |  |         path_append(jq, key); | 
					
						
							|  |  |  |         stack_push(jq, value); | 
					
						
							| 
									
										
										
										
											2012-08-16 01:00:30 +01:00
										 |  |  |       } | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     do_backtrack: | 
					
						
							|  |  |  |     case BACKTRACK: { | 
					
						
							| 
									
										
										
										
											2013-05-13 15:37:57 +01:00
										 |  |  |       pc = stack_restore(jq); | 
					
						
							|  |  |  |       if (!pc) { | 
					
						
							| 
									
										
										
										
											2014-07-05 20:54:42 -05:00
										 |  |  |         if (!jv_is_valid(jq->error)) { | 
					
						
							|  |  |  |           jv error = jq->error; | 
					
						
							|  |  |  |           jq->error = jv_null(); | 
					
						
							|  |  |  |           return error; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2012-09-03 16:16:14 +01:00
										 |  |  |         return jv_invalid(); | 
					
						
							| 
									
										
										
										
											2012-08-16 01:00:30 +01:00
										 |  |  |       } | 
					
						
							|  |  |  |       backtracking = 1; | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-07-05 20:54:42 -05:00
										 |  |  |     case FORK_OPT: | 
					
						
							| 
									
										
										
										
											2012-08-16 01:00:30 +01:00
										 |  |  |     case FORK: { | 
					
						
							| 
									
										
										
										
											2013-06-13 21:50:32 +01:00
										 |  |  |       stack_save(jq, pc - 1, stack_get_pos(jq)); | 
					
						
							| 
									
										
										
										
											2012-08-16 01:00:30 +01:00
										 |  |  |       pc++; // skip offset this time
 | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2012-08-27 10:11:55 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-07-05 20:54:42 -05:00
										 |  |  |     case ON_BACKTRACK(FORK_OPT): { | 
					
						
							|  |  |  |       if (jv_is_valid(jq->error)) { | 
					
						
							|  |  |  |         // `try EXP ...` backtracked here (no value, `empty`), so we backtrack more
 | 
					
						
							|  |  |  |         jv_free(stack_pop(jq)); | 
					
						
							|  |  |  |         goto do_backtrack; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       // `try EXP ...` exception caught in EXP
 | 
					
						
							|  |  |  |       jv_free(stack_pop(jq)); // free the input
 | 
					
						
							|  |  |  |       stack_push(jq, jv_invalid_get_msg(jq->error));  // push the error's message
 | 
					
						
							|  |  |  |       jq->error = jv_null(); | 
					
						
							| 
									
										
										
										
											2014-07-10 19:20:58 -05:00
										 |  |  |       uint16_t offset = *pc++; | 
					
						
							|  |  |  |       pc += offset; | 
					
						
							|  |  |  |       break; | 
					
						
							| 
									
										
										
										
											2014-07-05 20:54:42 -05:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2012-08-16 01:00:30 +01:00
										 |  |  |     case ON_BACKTRACK(FORK): { | 
					
						
							| 
									
										
										
										
											2014-07-10 19:20:58 -05:00
										 |  |  |       if (raising) goto do_backtrack; | 
					
						
							| 
									
										
										
										
											2012-08-16 01:00:30 +01:00
										 |  |  |       uint16_t offset = *pc++; | 
					
						
							|  |  |  |       pc += offset; | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |        | 
					
						
							| 
									
										
										
										
											2012-12-02 20:45:55 +00:00
										 |  |  |     case CALL_BUILTIN: { | 
					
						
							|  |  |  |       int nargs = *pc++; | 
					
						
							| 
									
										
										
										
											2013-05-13 15:00:05 +01:00
										 |  |  |       jv top = stack_pop(jq); | 
					
						
							| 
									
										
										
										
											2013-06-21 12:06:28 +01:00
										 |  |  |       jv* in = cfunc_input; | 
					
						
							|  |  |  |       in[0] = top; | 
					
						
							| 
									
										
										
										
											2012-12-02 20:45:55 +00:00
										 |  |  |       for (int i = 1; i < nargs; i++) { | 
					
						
							| 
									
										
										
										
											2013-06-21 12:06:28 +01:00
										 |  |  |         in[i] = stack_pop(jq); | 
					
						
							| 
									
										
										
										
											2012-09-10 16:07:03 +01:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2013-06-21 12:06:28 +01:00
										 |  |  |       struct cfunction* function = &frame_current(jq)->bc->globals->cfunctions[*pc++]; | 
					
						
							| 
									
										
										
										
											2014-07-07 22:49:46 -05:00
										 |  |  |       typedef jv (*func_1)(jq_state*,jv); | 
					
						
							|  |  |  |       typedef jv (*func_2)(jq_state*,jv,jv); | 
					
						
							|  |  |  |       typedef jv (*func_3)(jq_state*,jv,jv,jv); | 
					
						
							|  |  |  |       typedef jv (*func_4)(jq_state*,jv,jv,jv,jv); | 
					
						
							|  |  |  |       typedef jv (*func_5)(jq_state*,jv,jv,jv,jv,jv); | 
					
						
							| 
									
										
										
										
											2013-06-21 12:06:28 +01:00
										 |  |  |       switch (function->nargs) { | 
					
						
							| 
									
										
										
										
											2014-07-07 22:49:46 -05:00
										 |  |  |       case 1: top = ((func_1)function->fptr)(jq, in[0]); break; | 
					
						
							|  |  |  |       case 2: top = ((func_2)function->fptr)(jq, in[0], in[1]); break; | 
					
						
							|  |  |  |       case 3: top = ((func_3)function->fptr)(jq, in[0], in[1], in[2]); break; | 
					
						
							|  |  |  |       case 4: top = ((func_4)function->fptr)(jq, in[0], in[1], in[2], in[3]); break; | 
					
						
							|  |  |  |       case 5: top = ((func_5)function->fptr)(jq, in[0], in[1], in[2], in[3], in[4]); break; | 
					
						
							| 
									
										
										
										
											2014-07-05 20:54:42 -05:00
										 |  |  |       // FIXME: a) up to 7 arguments (input + 6), b) should assert
 | 
					
						
							|  |  |  |       // because the compiler should not generate this error.
 | 
					
						
							| 
									
										
										
										
											2013-06-21 12:06:28 +01:00
										 |  |  |       default: return jv_invalid_with_msg(jv_string("Function takes too many arguments")); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |        | 
					
						
							| 
									
										
										
										
											2013-05-13 15:00:05 +01:00
										 |  |  |       if (jv_is_valid(top)) { | 
					
						
							| 
									
										
										
										
											2013-04-28 18:33:45 -05:00
										 |  |  |         stack_push(jq, top); | 
					
						
							| 
									
										
										
										
											2014-12-26 14:37:38 -06:00
										 |  |  |       } else if (jv_invalid_has_msg(jv_copy(top))) { | 
					
						
							| 
									
										
										
										
											2014-08-11 17:25:09 -05:00
										 |  |  |         set_error(jq, top); | 
					
						
							| 
									
										
										
										
											2012-09-10 16:07:03 +01:00
										 |  |  |         goto do_backtrack; | 
					
						
							| 
									
										
										
										
											2014-12-26 14:37:38 -06:00
										 |  |  |       } else { | 
					
						
							|  |  |  |         // C-coded function returns invalid w/o msg? -> backtrack, as if
 | 
					
						
							|  |  |  |         // it had returned `empty`
 | 
					
						
							|  |  |  |         goto do_backtrack; | 
					
						
							| 
									
										
										
										
											2012-09-10 16:07:03 +01:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2012-08-16 01:00:30 +01:00
										 |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2012-08-21 12:35:36 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-30 23:41:20 -05:00
										 |  |  |     case TAIL_CALL_JQ: | 
					
						
							| 
									
										
										
										
											2012-12-02 20:45:55 +00:00
										 |  |  |     case CALL_JQ: { | 
					
						
							| 
									
										
										
										
											2014-06-30 19:40:56 -05:00
										 |  |  |       /*
 | 
					
						
							|  |  |  |        * Bytecode layout here: | 
					
						
							|  |  |  |        * | 
					
						
							|  |  |  |        *  CALL_JQ | 
					
						
							|  |  |  |        *  <nclosures>                       (i.e., number of call arguments) | 
					
						
							|  |  |  |        *  <callee closure>                  (what we're calling) | 
					
						
							|  |  |  |        *  <nclosures' worth of closures>    (frame reference + code pointer) | 
					
						
							|  |  |  |        * | 
					
						
							|  |  |  |        *  <next instruction (to return to)> | 
					
						
							|  |  |  |        * | 
					
						
							|  |  |  |        * Each closure consists of two uint16_t values: a "level" | 
					
						
							|  |  |  |        * identifying the frame to be closed over, and an index. | 
					
						
							|  |  |  |        * | 
					
						
							|  |  |  |        * The level is a relative number of call frames reachable from | 
					
						
							|  |  |  |        * the currently one; 0 -> current frame, 1 -> previous frame, and | 
					
						
							|  |  |  |        * so on. | 
					
						
							|  |  |  |        * | 
					
						
							|  |  |  |        * The index is either an index of the closed frame's subfunctions | 
					
						
							|  |  |  |        * or of the closed frame's parameter closures.  If the latter, | 
					
						
							|  |  |  |        * that closure will be passed, else the closed frame's pointer | 
					
						
							|  |  |  |        * and the subfunction's code will form the closure to be passed. | 
					
						
							|  |  |  |        * | 
					
						
							|  |  |  |        * See make_closure() for more information. | 
					
						
							|  |  |  |        */ | 
					
						
							| 
									
										
										
										
											2013-06-10 01:17:58 +01:00
										 |  |  |       jv input = stack_pop(jq); | 
					
						
							| 
									
										
										
										
											2012-08-21 18:14:13 +01:00
										 |  |  |       uint16_t nclosures = *pc++; | 
					
						
							| 
									
										
										
										
											2014-06-30 19:40:56 -05:00
										 |  |  |       uint16_t* retaddr = pc + 2 + nclosures*2; | 
					
						
							|  |  |  |       stack_ptr retdata = jq->stk_top; | 
					
						
							| 
									
										
										
										
											2014-06-22 00:24:02 -05:00
										 |  |  |       struct frame* new_frame; | 
					
						
							|  |  |  |       struct closure cl = make_closure(jq, pc); | 
					
						
							| 
									
										
										
										
											2014-06-30 23:41:20 -05:00
										 |  |  |       if (opcode == TAIL_CALL_JQ) { | 
					
						
							| 
									
										
										
										
											2014-06-22 00:24:02 -05:00
										 |  |  |         retaddr = frame_current(jq)->retaddr; | 
					
						
							| 
									
										
										
										
											2014-06-30 19:40:56 -05:00
										 |  |  |         retdata = frame_current(jq)->retdata; | 
					
						
							| 
									
										
										
										
											2014-06-22 00:24:02 -05:00
										 |  |  |         frame_pop(jq); | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2014-06-30 19:40:56 -05:00
										 |  |  |       new_frame = frame_push(jq, cl, pc + 2, nclosures); | 
					
						
							|  |  |  |       new_frame->retdata = retdata; | 
					
						
							| 
									
										
										
										
											2013-06-14 22:08:18 +01:00
										 |  |  |       new_frame->retaddr = retaddr; | 
					
						
							|  |  |  |       pc = new_frame->bc->code; | 
					
						
							| 
									
										
										
										
											2013-06-10 01:17:58 +01:00
										 |  |  |       stack_push(jq, input); | 
					
						
							| 
									
										
										
										
											2012-08-21 18:14:13 +01:00
										 |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case RET: { | 
					
						
							| 
									
										
										
										
											2013-06-10 01:17:58 +01:00
										 |  |  |       jv value = stack_pop(jq); | 
					
						
							| 
									
										
										
										
											2013-06-14 22:08:18 +01:00
										 |  |  |       assert(jq->stk_top == frame_current(jq)->retdata); | 
					
						
							|  |  |  |       uint16_t* retaddr = frame_current(jq)->retaddr; | 
					
						
							| 
									
										
										
										
											2013-05-15 01:23:06 +01:00
										 |  |  |       if (retaddr) { | 
					
						
							|  |  |  |         // function return
 | 
					
						
							|  |  |  |         pc = retaddr; | 
					
						
							| 
									
										
										
										
											2013-06-14 22:08:18 +01:00
										 |  |  |         frame_pop(jq); | 
					
						
							| 
									
										
										
										
											2013-05-15 01:23:06 +01:00
										 |  |  |       } else { | 
					
						
							|  |  |  |         // top-level return, yielding value
 | 
					
						
							| 
									
										
										
										
											2013-06-13 21:50:32 +01:00
										 |  |  |         struct stack_pos spos = stack_get_pos(jq); | 
					
						
							| 
									
										
										
										
											2013-05-15 01:23:06 +01:00
										 |  |  |         stack_push(jq, jv_null()); | 
					
						
							| 
									
										
										
										
											2013-06-13 21:50:32 +01:00
										 |  |  |         stack_save(jq, pc - 1, spos); | 
					
						
							| 
									
										
										
										
											2013-05-15 01:23:06 +01:00
										 |  |  |         return value; | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2013-06-10 01:17:58 +01:00
										 |  |  |       stack_push(jq, value); | 
					
						
							| 
									
										
										
										
											2012-08-21 18:14:13 +01:00
										 |  |  |       break; | 
					
						
							| 
									
										
										
										
											2012-08-21 12:35:36 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2013-05-15 01:23:06 +01:00
										 |  |  |     case ON_BACKTRACK(RET): { | 
					
						
							|  |  |  |       // resumed after top-level return
 | 
					
						
							|  |  |  |       goto do_backtrack; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2012-08-16 01:00:30 +01:00
										 |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-08-11 17:25:09 -05:00
										 |  |  | jv jq_format_error(jv msg) { | 
					
						
							|  |  |  |   if (jv_get_kind(msg) == JV_KIND_NULL || | 
					
						
							|  |  |  |       (jv_get_kind(msg) == JV_KIND_INVALID && !jv_invalid_has_msg(jv_copy(msg)))) { | 
					
						
							|  |  |  |     jv_free(msg); | 
					
						
							|  |  |  |     fprintf(stderr, "jq: error: out of memory\n"); | 
					
						
							|  |  |  |     return jv_null(); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (jv_get_kind(msg) == JV_KIND_STRING) | 
					
						
							|  |  |  |     return msg;                         // expected to already be formatted
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (jv_get_kind(msg) == JV_KIND_INVALID) | 
					
						
							|  |  |  |     msg = jv_invalid_get_msg(msg); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (jv_get_kind(msg) == JV_KIND_NULL) | 
					
						
							|  |  |  |     return jq_format_error(msg);        // ENOMEM
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Invalid with msg; prefix with "jq: error: "
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (jv_get_kind(msg) != JV_KIND_INVALID) { | 
					
						
							|  |  |  |     if (jv_get_kind(msg) == JV_KIND_STRING) | 
					
						
							| 
									
										
										
										
											2014-12-23 21:52:03 -06:00
										 |  |  |       return jv_string_fmt("jq: error: %s", jv_string_value(msg)); | 
					
						
							| 
									
										
										
										
											2014-08-11 17:25:09 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     msg = jv_dump_string(msg, JV_PRINT_INVALID); | 
					
						
							|  |  |  |     if (jv_get_kind(msg) == JV_KIND_STRING) | 
					
						
							| 
									
										
										
										
											2014-12-23 21:52:03 -06:00
										 |  |  |       return jv_string_fmt("jq: error: %s", jv_string_value(msg)); | 
					
						
							| 
									
										
										
										
											2014-08-11 17:25:09 -05:00
										 |  |  |     return jq_format_error(jv_null());  // ENOMEM
 | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // An invalid inside an invalid!
 | 
					
						
							|  |  |  |   return jq_format_error(jv_invalid_get_msg(msg)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // XXX Refactor into a utility function that returns a jv and one that
 | 
					
						
							|  |  |  | // uses it and then prints that jv's string as the complete error
 | 
					
						
							|  |  |  | // message.
 | 
					
						
							|  |  |  | static void default_err_cb(void *data, jv msg) { | 
					
						
							|  |  |  |   msg = jq_format_error(msg); | 
					
						
							|  |  |  |   fprintf((FILE *)data, "%s\n", jv_string_value(msg)); | 
					
						
							|  |  |  |   jv_free(msg); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-06 17:26:15 -05:00
										 |  |  | jq_state *jq_init(void) { | 
					
						
							|  |  |  |   jq_state *jq; | 
					
						
							|  |  |  |   jq = jv_mem_alloc_unguarded(sizeof(*jq)); | 
					
						
							|  |  |  |   if (jq == NULL) | 
					
						
							|  |  |  |     return NULL; | 
					
						
							| 
									
										
										
										
											2013-06-18 00:17:20 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |   jq->bc = 0; | 
					
						
							| 
									
										
										
										
											2013-06-18 00:06:00 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |   stack_init(&jq->stk); | 
					
						
							|  |  |  |   jq->stk_top = 0; | 
					
						
							|  |  |  |   jq->fork_top = 0; | 
					
						
							|  |  |  |   jq->curr_frame = 0; | 
					
						
							| 
									
										
										
										
											2014-07-05 20:54:42 -05:00
										 |  |  |   jq->error = jv_null(); | 
					
						
							| 
									
										
										
										
											2013-06-18 00:06:00 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-08-11 17:25:09 -05:00
										 |  |  |   jq->err_cb = default_err_cb; | 
					
						
							|  |  |  |   jq->err_cb_data = stderr; | 
					
						
							| 
									
										
										
										
											2013-11-30 02:05:42 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-07-09 00:55:20 -04:00
										 |  |  |   jq->attrs = jv_object(); | 
					
						
							| 
									
										
										
										
											2013-06-06 17:26:15 -05:00
										 |  |  |   jq->path = jv_null(); | 
					
						
							|  |  |  |   return jq; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-30 02:05:42 -06:00
										 |  |  | void jq_set_error_cb(jq_state *jq, jq_err_cb cb, void *data) { | 
					
						
							| 
									
										
										
										
											2014-08-11 17:25:09 -05:00
										 |  |  |   assert(cb != NULL); | 
					
						
							| 
									
										
										
										
											2013-11-30 02:05:42 -06:00
										 |  |  |   jq->err_cb = cb; | 
					
						
							|  |  |  |   jq->err_cb_data = data; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void jq_get_error_cb(jq_state *jq, jq_err_cb *cb, void **data) { | 
					
						
							|  |  |  |   *cb = jq->err_cb; | 
					
						
							|  |  |  |   *data = jq->err_cb_data; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-06 17:26:15 -05:00
										 |  |  | void jq_set_nomem_handler(jq_state *jq, void (*nomem_handler)(void *), void *data) { | 
					
						
							|  |  |  |   jv_nomem_handler(nomem_handler, data); | 
					
						
							|  |  |  |   jq->nomem_handler = nomem_handler; | 
					
						
							|  |  |  |   jq->nomem_handler_data = data; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2012-08-16 01:00:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-18 00:06:00 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-06 17:26:15 -05:00
										 |  |  | void jq_start(jq_state *jq, jv input, int flags) { | 
					
						
							|  |  |  |   jv_nomem_handler(jq->nomem_handler, jq->nomem_handler_data); | 
					
						
							|  |  |  |   jq_reset(jq); | 
					
						
							| 
									
										
										
										
											2012-08-16 01:00:30 +01:00
										 |  |  |    | 
					
						
							| 
									
										
										
										
											2013-06-06 17:26:15 -05:00
										 |  |  |   struct closure top = {jq->bc, -1}; | 
					
						
							| 
									
										
										
										
											2013-06-18 00:06:00 +01:00
										 |  |  |   struct frame* top_frame = frame_push(jq, top, 0, 0); | 
					
						
							| 
									
										
										
										
											2013-06-14 22:08:18 +01:00
										 |  |  |   top_frame->retdata = 0; | 
					
						
							|  |  |  |   top_frame->retaddr = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-18 00:06:00 +01:00
										 |  |  |   stack_push(jq, input); | 
					
						
							|  |  |  |   stack_save(jq, jq->bc->code, stack_get_pos(jq)); | 
					
						
							| 
									
										
										
										
											2013-05-05 23:12:10 +01:00
										 |  |  |   if (flags & JQ_DEBUG_TRACE) { | 
					
						
							| 
									
										
										
										
											2013-06-06 17:26:15 -05:00
										 |  |  |     jq->debug_trace_enabled = 1; | 
					
						
							| 
									
										
										
										
											2013-05-05 23:12:10 +01:00
										 |  |  |   } else { | 
					
						
							| 
									
										
										
										
											2013-06-06 17:26:15 -05:00
										 |  |  |     jq->debug_trace_enabled = 0; | 
					
						
							| 
									
										
										
										
											2013-05-05 23:12:10 +01:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2013-06-06 17:26:15 -05:00
										 |  |  |   jq->initial_execution = 1; | 
					
						
							| 
									
										
										
										
											2012-08-16 01:00:30 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-28 18:33:45 -05:00
										 |  |  | void jq_teardown(jq_state **jq) { | 
					
						
							|  |  |  |   jq_state *old_jq = *jq; | 
					
						
							|  |  |  |   if (old_jq == NULL) | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   *jq = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-06 17:26:15 -05:00
										 |  |  |   jq_reset(old_jq); | 
					
						
							|  |  |  |   bytecode_free(old_jq->bc); | 
					
						
							|  |  |  |   old_jq->bc = 0; | 
					
						
							| 
									
										
										
										
											2014-07-09 00:55:20 -04:00
										 |  |  |   jv_free(old_jq->attrs); | 
					
						
							| 
									
										
										
										
											2012-09-03 15:33:59 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-28 18:33:45 -05:00
										 |  |  |   jv_mem_free(old_jq); | 
					
						
							| 
									
										
										
										
											2012-09-02 21:45:27 +01:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2012-09-18 22:17:13 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-30 23:41:20 -05:00
										 |  |  | static int ret_follows(uint16_t *pc) { | 
					
						
							|  |  |  |   if (*pc == RET) | 
					
						
							|  |  |  |     return 1; | 
					
						
							|  |  |  |   if (*pc++ != JUMP) | 
					
						
							|  |  |  |     return 0; | 
					
						
							| 
									
										
										
										
											2014-07-01 22:40:40 -05:00
										 |  |  |   return ret_follows(pc + *pc + 1); // FIXME, might be ironic
 | 
					
						
							| 
									
										
										
										
											2014-06-30 23:41:20 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * Look for tail calls that can be optimized: tail calls with no | 
					
						
							|  |  |  |  * references left to the current frame. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * We're staring at this bytecode layout: | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *   CALL_JQ | 
					
						
							|  |  |  |  *   <nclosures> | 
					
						
							|  |  |  |  *   <callee closure>       (2 units) | 
					
						
							|  |  |  |  *   <nclosures closures>   (2 units each) | 
					
						
							|  |  |  |  *   <next instruction> | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * A closure is: | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *   <level>    (a relative frame count chased via the current frame's env) | 
					
						
							|  |  |  |  *   <index>    (an index of a subfunction or closure in that frame) | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * We're looking for: | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * a) the next instruction is a RET or a chain of unconditional JUMPs | 
					
						
							|  |  |  |  * that ends in a RET, and | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * b) none of the closures -callee included- have level == 0. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static uint16_t tail_call_analyze(uint16_t *pc) { | 
					
						
							|  |  |  |   assert(*pc == CALL_JQ); | 
					
						
							|  |  |  |   pc++; | 
					
						
							|  |  |  |   // + 1 for the callee closure
 | 
					
						
							|  |  |  |   for (uint16_t nclosures = *pc++ + 1; nclosures > 0; pc++, nclosures--) { | 
					
						
							|  |  |  |     if (*pc++ == 0) | 
					
						
							|  |  |  |       return CALL_JQ; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   if (ret_follows(pc)) | 
					
						
							|  |  |  |     return TAIL_CALL_JQ; | 
					
						
							|  |  |  |   return CALL_JQ; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct bytecode *optimize_code(struct bytecode *bc) { | 
					
						
							|  |  |  |   uint16_t *pc = bc->code; | 
					
						
							|  |  |  |   // FIXME: Don't mutate bc->code...
 | 
					
						
							|  |  |  |   while (pc < bc->code + bc->codelen) { | 
					
						
							|  |  |  |     switch (*pc) { | 
					
						
							|  |  |  |     case CALL_JQ: | 
					
						
							|  |  |  |       *pc = tail_call_analyze(pc); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Other bytecode optimizations here.  A peephole optimizer would
 | 
					
						
							|  |  |  |     // fit right in.
 | 
					
						
							|  |  |  |     default: break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     pc += bytecode_operation_length(pc); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return bc; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct bytecode *optimize(struct bytecode *bc) { | 
					
						
							|  |  |  |   for (int i=0; i<bc->nsubfunctions; i++) { | 
					
						
							|  |  |  |     bc->subfunctions[i] = optimize(bc->subfunctions[i]); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return optimize_code(bc); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-06 17:26:15 -05:00
										 |  |  | int jq_compile_args(jq_state *jq, const char* str, jv args) { | 
					
						
							|  |  |  |   jv_nomem_handler(jq->nomem_handler, jq->nomem_handler_data); | 
					
						
							| 
									
										
										
										
											2013-05-06 14:21:00 +01:00
										 |  |  |   assert(jv_get_kind(args) == JV_KIND_ARRAY); | 
					
						
							| 
									
										
										
										
											2014-07-09 00:55:20 -04:00
										 |  |  |   struct locfile* locations; | 
					
						
							|  |  |  |   locations = locfile_init(jq, str, strlen(str)); | 
					
						
							| 
									
										
										
										
											2012-09-18 22:17:13 +01:00
										 |  |  |   block program; | 
					
						
							| 
									
										
										
										
											2013-06-06 17:26:15 -05:00
										 |  |  |   jq_reset(jq); | 
					
						
							|  |  |  |   if (jq->bc) { | 
					
						
							|  |  |  |     bytecode_free(jq->bc); | 
					
						
							|  |  |  |     jq->bc = 0; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2014-07-09 00:55:20 -04:00
										 |  |  |   int nerrors = load_program(jq, locations, &program); | 
					
						
							| 
									
										
										
										
											2012-09-18 22:17:13 +01:00
										 |  |  |   if (nerrors == 0) { | 
					
						
							| 
									
										
										
										
											2014-07-09 00:55:20 -04:00
										 |  |  |     jv_array_foreach(args, i, arg) { | 
					
						
							| 
									
										
										
										
											2013-05-06 14:21:00 +01:00
										 |  |  |       jv name = jv_object_get(jv_copy(arg), jv_string("name")); | 
					
						
							|  |  |  |       jv value = jv_object_get(arg, jv_string("value")); | 
					
						
							|  |  |  |       program = gen_var_binding(gen_const(value), jv_string_value(name), program); | 
					
						
							|  |  |  |       jv_free(name); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2014-07-09 00:55:20 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-30 02:05:42 -06:00
										 |  |  |     nerrors = builtins_bind(jq, &program); | 
					
						
							| 
									
										
										
										
											2013-05-17 03:03:42 +10:00
										 |  |  |     if (nerrors == 0) { | 
					
						
							| 
									
										
										
										
											2014-07-09 00:55:20 -04:00
										 |  |  |       nerrors = block_compile(program, &jq->bc); | 
					
						
							| 
									
										
										
										
											2013-05-17 03:03:42 +10:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2012-09-18 22:17:13 +01:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2014-08-11 17:25:09 -05:00
										 |  |  |   if (nerrors) | 
					
						
							|  |  |  |     jq_report_error(jq, jv_string_fmt("jq: %d compile %s", nerrors, nerrors > 1 ? "errors" : "error")); | 
					
						
							| 
									
										
										
										
											2014-06-30 23:41:20 -05:00
										 |  |  |   if (jq->bc) | 
					
						
							|  |  |  |     jq->bc = optimize(jq->bc); | 
					
						
							| 
									
										
										
										
											2014-07-09 00:55:20 -04:00
										 |  |  |   jv_free(args); | 
					
						
							|  |  |  |   locfile_free(locations); | 
					
						
							| 
									
										
										
										
											2013-06-06 17:26:15 -05:00
										 |  |  |   return jq->bc != NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int jq_compile(jq_state *jq, const char* str) { | 
					
						
							|  |  |  |   return jq_compile_args(jq, str, jv_array()); | 
					
						
							| 
									
										
										
										
											2012-09-18 22:17:13 +01:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2013-05-06 14:21:00 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-24 17:58:34 -06:00
										 |  |  | jv jq_get_jq_origin(jq_state *jq) { | 
					
						
							|  |  |  |   return jq_get_attr(jq, jv_string("JQ_ORIGIN")); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | jv jq_get_prog_origin(jq_state *jq) { | 
					
						
							|  |  |  |   return jq_get_attr(jq, jv_string("PROGRAM_ORIGIN")); | 
					
						
							| 
									
										
										
										
											2014-07-09 00:55:20 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | jv jq_get_lib_dirs(jq_state *jq) { | 
					
						
							| 
									
										
										
										
											2014-11-24 17:58:34 -06:00
										 |  |  |   return jq_get_attr(jq, jv_string("JQ_LIBRARY_PATH")); | 
					
						
							| 
									
										
										
										
											2014-08-10 16:52:03 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void jq_set_attrs(jq_state *jq, jv attrs) { | 
					
						
							|  |  |  |   assert(jv_get_kind(attrs) == JV_KIND_OBJECT); | 
					
						
							|  |  |  |   jv_free(jq->attrs); | 
					
						
							|  |  |  |   jq->attrs = attrs; | 
					
						
							| 
									
										
										
										
											2014-07-09 00:55:20 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void jq_set_attr(jq_state *jq, jv attr, jv val) { | 
					
						
							| 
									
										
										
										
											2014-08-10 16:52:03 -05:00
										 |  |  |   jq->attrs = jv_object_set(jq->attrs, attr, val); | 
					
						
							| 
									
										
										
										
											2014-07-09 00:55:20 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | jv jq_get_attr(jq_state *jq, jv attr) { | 
					
						
							| 
									
										
										
										
											2014-08-10 16:52:03 -05:00
										 |  |  |   return jv_object_get(jv_copy(jq->attrs), attr); | 
					
						
							| 
									
										
										
										
											2014-07-09 00:55:20 -04:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2014-08-10 16:52:03 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-06 17:26:15 -05:00
										 |  |  | void jq_dump_disassembly(jq_state *jq, int indent) { | 
					
						
							|  |  |  |   dump_disassembly(indent, jq->bc); | 
					
						
							| 
									
										
										
										
											2013-05-06 14:21:00 +01:00
										 |  |  | } | 
					
						
							| 
									
										
										
											
												Add Streaming parser (--stream)
Streaming means that outputs are produced as soon as possible.  With the
`foreach` syntax one can write programs which reduce portions of the
streaming parse of a large input (reduce into proper JSON values, for
example), and discard the rest, processing incrementally.
This:
    $ jq -c --stream .
should produce the same output as this:
    $ jq -c '. as $dot | path(..) as $p | $dot | getpath($p) | [$p,.]'
The output of `jq --stream .` should be a sequence of`[[<path>],<leaf>]`
and `[[<path>]]` values.  The latter indicate that the array/object at
that path ended.
Scalars and empty arrays and objects are leaf values for this purpose.
For example, a truncated input produces a path as soon as possible, then
later the error:
    $ printf '[0,\n'|./jq -c --stream .
    [[0],0]
    parse error: Unfinished JSON term at EOF at line 3, column 0
    $
											
										 
											2014-12-22 23:06:27 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  | void jq_set_input_cb(jq_state *jq, jq_input_cb cb, void *data) { | 
					
						
							|  |  |  |   jq->input_cb = cb; | 
					
						
							|  |  |  |   jq->input_cb_data = data; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void jq_get_input_cb(jq_state *jq, jq_input_cb *cb, void **data) { | 
					
						
							|  |  |  |   *cb = jq->input_cb; | 
					
						
							|  |  |  |   *data = jq->input_cb_data; | 
					
						
							|  |  |  | } |