mirror of
https://github.com/MariaDB/server.git
synced 2025-01-23 23:34:34 +01:00
7197 lines
245 KiB
C
7197 lines
245 KiB
C
/* -*- c-basic-offset: 2 -*- */
|
|
/*
|
|
Copyright(C) 2010-2015 Brazil
|
|
|
|
This library is free software; you can redistribute it and/or
|
|
modify it under the terms of the GNU Lesser General Public
|
|
License version 2.1 as published by the Free Software Foundation.
|
|
|
|
This library is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
Lesser General Public License for more details.
|
|
|
|
You should have received a copy of the GNU Lesser General Public
|
|
License along with this library; if not, write to the Free Software
|
|
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
|
*/
|
|
#include "grn.h"
|
|
#include "grn_db.h"
|
|
#include "grn_ctx_impl.h"
|
|
#include <string.h>
|
|
#include <float.h>
|
|
#include "grn_ii.h"
|
|
#include "grn_geo.h"
|
|
#include "grn_expr.h"
|
|
#include "grn_util.h"
|
|
#include "grn_normalizer.h"
|
|
#include "grn_mrb.h"
|
|
#include "mrb/mrb_expr.h"
|
|
|
|
grn_obj *
|
|
grn_expr_alloc(grn_ctx *ctx, grn_obj *expr, grn_id domain, grn_obj_flags flags)
|
|
{
|
|
grn_obj *res = NULL;
|
|
grn_expr *e = (grn_expr *)expr;
|
|
if (e) {
|
|
if (e->values_curr >= e->values_size) {
|
|
// todo : expand values.
|
|
ERR(GRN_NO_MEMORY_AVAILABLE, "no more e->values");
|
|
return NULL;
|
|
}
|
|
res = &e->values[e->values_curr++];
|
|
if (e->values_curr > e->values_tail) { e->values_tail = e->values_curr; }
|
|
grn_obj_reinit(ctx, res, domain, flags);
|
|
}
|
|
return res;
|
|
}
|
|
|
|
grn_hash *
|
|
grn_expr_get_vars(grn_ctx *ctx, grn_obj *expr, unsigned int *nvars)
|
|
{
|
|
grn_hash *vars = NULL;
|
|
if (expr->header.type == GRN_PROC || expr->header.type == GRN_EXPR) {
|
|
grn_id id = DB_OBJ(expr)->id;
|
|
grn_expr *e = (grn_expr *)expr;
|
|
int added = 0;
|
|
grn_hash **vp;
|
|
if (grn_hash_add(ctx, ctx->impl->expr_vars, &id, sizeof(grn_id), (void **)&vp, &added)) {
|
|
if (!*vp) {
|
|
*vp = grn_hash_create(ctx, NULL, GRN_TABLE_MAX_KEY_SIZE, sizeof(grn_obj),
|
|
GRN_OBJ_KEY_VAR_SIZE|GRN_OBJ_TEMPORARY|GRN_HASH_TINY);
|
|
if (*vp) {
|
|
uint32_t i;
|
|
grn_obj *value;
|
|
grn_expr_var *v;
|
|
for (v = e->vars, i = e->nvars; i; v++, i--) {
|
|
grn_hash_add(ctx, *vp, v->name, v->name_size, (void **)&value, &added);
|
|
GRN_OBJ_INIT(value, v->value.header.type, 0, v->value.header.domain);
|
|
GRN_TEXT_PUT(ctx, value, GRN_TEXT_VALUE(&v->value), GRN_TEXT_LEN(&v->value));
|
|
}
|
|
}
|
|
}
|
|
vars = *vp;
|
|
}
|
|
}
|
|
*nvars = vars ? GRN_HASH_SIZE(vars) : 0;
|
|
return vars;
|
|
}
|
|
|
|
grn_rc
|
|
grn_expr_clear_vars(grn_ctx *ctx, grn_obj *expr)
|
|
{
|
|
if (expr->header.type == GRN_PROC || expr->header.type == GRN_EXPR) {
|
|
grn_hash **vp;
|
|
grn_id eid, id = DB_OBJ(expr)->id;
|
|
if ((eid = grn_hash_get(ctx, ctx->impl->expr_vars, &id, sizeof(grn_id), (void **)&vp))) {
|
|
if (*vp) {
|
|
grn_obj *value;
|
|
GRN_HASH_EACH(ctx, *vp, i, NULL, NULL, (void **)&value, {
|
|
GRN_OBJ_FIN(ctx, value);
|
|
});
|
|
grn_hash_close(ctx, *vp);
|
|
}
|
|
grn_hash_delete_by_id(ctx, ctx->impl->expr_vars, eid, NULL);
|
|
}
|
|
}
|
|
return ctx->rc;
|
|
}
|
|
|
|
grn_obj *
|
|
grn_proc_get_info(grn_ctx *ctx, grn_user_data *user_data,
|
|
grn_expr_var **vars, unsigned int *nvars, grn_obj **caller)
|
|
{
|
|
grn_proc_ctx *pctx = (grn_proc_ctx *)user_data;
|
|
if (caller) { *caller = pctx->caller; }
|
|
if (pctx->proc) {
|
|
if (vars) {
|
|
*vars = pctx->proc->vars;
|
|
// *vars = grn_expr_get_vars(ctx, (grn_obj *)pctx->proc, nvars);
|
|
}
|
|
if (nvars) { *nvars = pctx->proc->nvars; }
|
|
} else {
|
|
if (vars) { *vars = NULL; }
|
|
if (nvars) { *nvars = 0; }
|
|
}
|
|
return (grn_obj *)pctx->proc;
|
|
}
|
|
|
|
grn_obj *
|
|
grn_proc_get_vars(grn_ctx *ctx, grn_user_data *user_data)
|
|
{
|
|
uint32_t n;
|
|
grn_proc_ctx *pctx = (grn_proc_ctx *)user_data;
|
|
if (pctx->proc) {
|
|
return (grn_obj *)grn_expr_get_vars(ctx, (grn_obj *)pctx->proc, &n);
|
|
} else {
|
|
return NULL;
|
|
}
|
|
}
|
|
|
|
grn_obj *
|
|
grn_proc_get_var(grn_ctx *ctx, grn_user_data *user_data, const char *name, unsigned int name_size)
|
|
{
|
|
grn_proc_ctx *pctx = (grn_proc_ctx *)user_data;
|
|
return pctx->proc ? grn_expr_get_var(ctx, (grn_obj *)pctx->proc, name, name_size) : NULL;
|
|
}
|
|
|
|
grn_obj *
|
|
grn_proc_get_var_by_offset(grn_ctx *ctx, grn_user_data *user_data, unsigned int offset)
|
|
{
|
|
grn_proc_ctx *pctx = (grn_proc_ctx *)user_data;
|
|
return pctx->proc ? grn_expr_get_var_by_offset(ctx, (grn_obj *)pctx->proc, offset) : NULL;
|
|
}
|
|
|
|
grn_obj *
|
|
grn_proc_get_or_add_var(grn_ctx *ctx, grn_user_data *user_data,
|
|
const char *name, unsigned int name_size)
|
|
{
|
|
grn_proc_ctx *pctx = (grn_proc_ctx *)user_data;
|
|
return pctx->proc ? grn_expr_get_or_add_var(ctx, (grn_obj *)pctx->proc, name, name_size) : NULL;
|
|
}
|
|
|
|
grn_obj *
|
|
grn_proc_alloc(grn_ctx *ctx, grn_user_data *user_data, grn_id domain, grn_obj_flags flags)
|
|
{
|
|
grn_proc_ctx *pctx = (grn_proc_ctx *)user_data;
|
|
return pctx->caller ? grn_expr_alloc(ctx, (grn_obj *)pctx->caller, domain, flags) : NULL;
|
|
}
|
|
|
|
grn_proc_type
|
|
grn_proc_get_type(grn_ctx *ctx, grn_obj *proc)
|
|
{
|
|
grn_proc *proc_ = (grn_proc *)proc;
|
|
return proc_ ? proc_->type : GRN_PROC_INVALID;
|
|
}
|
|
|
|
grn_rc
|
|
grn_proc_set_selector(grn_ctx *ctx, grn_obj *proc, grn_selector_func selector)
|
|
{
|
|
grn_proc *proc_ = (grn_proc *)proc;
|
|
if (!grn_obj_is_function_proc(ctx, proc)) {
|
|
return GRN_INVALID_ARGUMENT;
|
|
}
|
|
proc_->selector = selector;
|
|
return GRN_SUCCESS;
|
|
}
|
|
|
|
/* grn_expr */
|
|
|
|
grn_obj *
|
|
grn_ctx_pop(grn_ctx *ctx)
|
|
{
|
|
if (ctx && ctx->impl && ctx->impl->stack_curr) {
|
|
return ctx->impl->stack[--ctx->impl->stack_curr];
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
grn_rc
|
|
grn_ctx_push(grn_ctx *ctx, grn_obj *obj)
|
|
{
|
|
if (ctx && ctx->impl && ctx->impl->stack_curr < GRN_STACK_SIZE) {
|
|
ctx->impl->stack[ctx->impl->stack_curr++] = obj;
|
|
return GRN_SUCCESS;
|
|
}
|
|
return GRN_STACK_OVER_FLOW;
|
|
}
|
|
|
|
grn_obj *
|
|
grn_expr_alloc_const(grn_ctx *ctx, grn_obj *expr)
|
|
{
|
|
grn_expr *e = (grn_expr *)expr;
|
|
|
|
if (!e->consts) {
|
|
if (!(e->consts = GRN_MALLOCN(grn_obj, GRN_STACK_SIZE))) {
|
|
ERR(GRN_NO_MEMORY_AVAILABLE, "malloc failed");
|
|
return NULL;
|
|
}
|
|
}
|
|
if (e->nconsts < GRN_STACK_SIZE) {
|
|
return e->consts + e->nconsts++;
|
|
} else {
|
|
ERR(GRN_STACK_OVER_FLOW, "too many constants.");
|
|
return NULL;
|
|
}
|
|
}
|
|
|
|
void
|
|
grn_obj_pack(grn_ctx *ctx, grn_obj *buf, grn_obj *obj)
|
|
{
|
|
grn_text_benc(ctx, buf, obj->header.type);
|
|
if (GRN_DB_OBJP(obj)) {
|
|
grn_text_benc(ctx, buf, DB_OBJ(obj)->id);
|
|
} else {
|
|
// todo : support vector, query, accessor, snip..
|
|
uint32_t vs = GRN_BULK_VSIZE(obj);
|
|
grn_text_benc(ctx, buf, obj->header.domain);
|
|
grn_text_benc(ctx, buf, vs);
|
|
if (vs) { GRN_TEXT_PUT(ctx, buf, GRN_BULK_HEAD(obj), vs); }
|
|
}
|
|
}
|
|
|
|
const uint8_t *
|
|
grn_obj_unpack(grn_ctx *ctx, const uint8_t *p, const uint8_t *pe, uint8_t type, uint8_t flags, grn_obj *obj)
|
|
{
|
|
grn_id domain;
|
|
uint32_t vs;
|
|
GRN_B_DEC(domain, p);
|
|
GRN_OBJ_INIT(obj, type, flags, domain);
|
|
GRN_B_DEC(vs, p);
|
|
if (pe < p + vs) {
|
|
ERR(GRN_INVALID_FORMAT, "benced image is corrupt");
|
|
return p;
|
|
}
|
|
grn_bulk_write(ctx, obj, (const char *)p, vs);
|
|
return p + vs;
|
|
}
|
|
|
|
typedef enum {
|
|
GRN_EXPR_PACK_TYPE_NULL = 0,
|
|
GRN_EXPR_PACK_TYPE_VARIABLE = 1,
|
|
GRN_EXPR_PACK_TYPE_OTHERS = 2
|
|
} grn_expr_pack_type;
|
|
|
|
void
|
|
grn_expr_pack(grn_ctx *ctx, grn_obj *buf, grn_obj *expr)
|
|
{
|
|
grn_expr_code *c;
|
|
grn_expr_var *v;
|
|
grn_expr *e = (grn_expr *)expr;
|
|
uint32_t i, j;
|
|
grn_text_benc(ctx, buf, e->nvars);
|
|
for (i = e->nvars, v = e->vars; i; i--, v++) {
|
|
grn_text_benc(ctx, buf, v->name_size);
|
|
if (v->name_size) { GRN_TEXT_PUT(ctx, buf, v->name, v->name_size); }
|
|
grn_obj_pack(ctx, buf, &v->value);
|
|
}
|
|
i = e->codes_curr;
|
|
grn_text_benc(ctx, buf, i);
|
|
for (c = e->codes; i; i--, c++) {
|
|
grn_text_benc(ctx, buf, c->op);
|
|
grn_text_benc(ctx, buf, c->nargs);
|
|
if (!c->value) {
|
|
grn_text_benc(ctx, buf, GRN_EXPR_PACK_TYPE_NULL);
|
|
} else {
|
|
for (j = 0, v = e->vars; j < e->nvars; j++, v++) {
|
|
if (&v->value == c->value) {
|
|
grn_text_benc(ctx, buf, GRN_EXPR_PACK_TYPE_VARIABLE);
|
|
grn_text_benc(ctx, buf, j);
|
|
break;
|
|
}
|
|
}
|
|
if (j == e->nvars) {
|
|
grn_text_benc(ctx, buf, GRN_EXPR_PACK_TYPE_OTHERS);
|
|
grn_obj_pack(ctx, buf, c->value);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
const uint8_t *
|
|
grn_expr_unpack(grn_ctx *ctx, const uint8_t *p, const uint8_t *pe, grn_obj *expr)
|
|
{
|
|
grn_obj *v;
|
|
grn_expr_pack_type type;
|
|
uint32_t i, n, ns;
|
|
grn_expr_code *code;
|
|
grn_expr *e = (grn_expr *)expr;
|
|
GRN_B_DEC(n, p);
|
|
for (i = 0; i < n; i++) {
|
|
uint32_t object_type;
|
|
GRN_B_DEC(ns, p);
|
|
v = grn_expr_add_var(ctx, expr, ns ? (const char *)p : NULL, ns);
|
|
p += ns;
|
|
GRN_B_DEC(object_type, p);
|
|
if (GRN_TYPE <= object_type && object_type <= GRN_COLUMN_INDEX) { /* error */ }
|
|
p = grn_obj_unpack(ctx, p, pe, object_type, 0, v);
|
|
if (pe < p) {
|
|
ERR(GRN_INVALID_FORMAT, "benced image is corrupt");
|
|
return p;
|
|
}
|
|
}
|
|
GRN_B_DEC(n, p);
|
|
/* confirm e->codes_size >= n */
|
|
e->codes_curr = n;
|
|
for (i = 0, code = e->codes; i < n; i++, code++) {
|
|
GRN_B_DEC(code->op, p);
|
|
GRN_B_DEC(code->nargs, p);
|
|
GRN_B_DEC(type, p);
|
|
switch (type) {
|
|
case GRN_EXPR_PACK_TYPE_NULL :
|
|
code->value = NULL;
|
|
break;
|
|
case GRN_EXPR_PACK_TYPE_VARIABLE :
|
|
{
|
|
uint32_t offset;
|
|
GRN_B_DEC(offset, p);
|
|
code->value = &e->vars[i].value;
|
|
}
|
|
break;
|
|
case GRN_EXPR_PACK_TYPE_OTHERS :
|
|
{
|
|
uint32_t object_type;
|
|
GRN_B_DEC(object_type, p);
|
|
if (GRN_TYPE <= object_type && object_type <= GRN_COLUMN_INDEX) {
|
|
grn_id id;
|
|
GRN_B_DEC(id, p);
|
|
code->value = grn_ctx_at(ctx, id);
|
|
} else {
|
|
if (!(v = grn_expr_alloc_const(ctx, expr))) { return NULL; }
|
|
p = grn_obj_unpack(ctx, p, pe, object_type, GRN_OBJ_EXPRCONST, v);
|
|
code->value = v;
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
if (pe < p) {
|
|
ERR(GRN_INVALID_FORMAT, "benced image is corrupt");
|
|
return p;
|
|
}
|
|
}
|
|
return p;
|
|
}
|
|
|
|
grn_obj *
|
|
grn_expr_open(grn_ctx *ctx, grn_obj_spec *spec, const uint8_t *p, const uint8_t *pe)
|
|
{
|
|
grn_expr *expr = NULL;
|
|
if ((expr = GRN_MALLOCN(grn_expr, 1))) {
|
|
int size = 256;
|
|
expr->consts = NULL;
|
|
expr->nconsts = 0;
|
|
GRN_TEXT_INIT(&expr->name_buf, 0);
|
|
GRN_TEXT_INIT(&expr->dfi, 0);
|
|
GRN_PTR_INIT(&expr->objs, GRN_OBJ_VECTOR, GRN_ID_NIL);
|
|
expr->vars = NULL;
|
|
expr->nvars = 0;
|
|
GRN_DB_OBJ_SET_TYPE(expr, GRN_EXPR);
|
|
if ((expr->values = GRN_MALLOCN(grn_obj, size))) {
|
|
int i;
|
|
for (i = 0; i < size; i++) {
|
|
GRN_OBJ_INIT(&expr->values[i], GRN_BULK, GRN_OBJ_EXPRVALUE, GRN_ID_NIL);
|
|
}
|
|
expr->values_curr = 0;
|
|
expr->values_tail = 0;
|
|
expr->values_size = size;
|
|
if ((expr->codes = GRN_MALLOCN(grn_expr_code, size))) {
|
|
expr->codes_curr = 0;
|
|
expr->codes_size = size;
|
|
expr->obj.header = spec->header;
|
|
if (grn_expr_unpack(ctx, p, pe, (grn_obj *)expr) == pe) {
|
|
goto exit;
|
|
} else {
|
|
ERR(GRN_INVALID_FORMAT, "benced image is corrupt");
|
|
}
|
|
GRN_FREE(expr->codes);
|
|
}
|
|
GRN_FREE(expr->values);
|
|
}
|
|
GRN_FREE(expr);
|
|
expr = NULL;
|
|
}
|
|
exit :
|
|
return (grn_obj *)expr;
|
|
}
|
|
|
|
/* Pass ownership of `obj` to `expr`. */
|
|
void
|
|
grn_expr_take_obj(grn_ctx *ctx, grn_obj *expr, grn_obj *obj)
|
|
{
|
|
grn_expr *e = (grn_expr *)expr;
|
|
GRN_PTR_PUT(ctx, &(e->objs), obj);
|
|
}
|
|
|
|
/* data flow info */
|
|
typedef struct {
|
|
grn_expr_code *code;
|
|
grn_id domain;
|
|
unsigned char type;
|
|
} grn_expr_dfi;
|
|
|
|
#define DFI_POP(e,d) do {\
|
|
if (GRN_BULK_VSIZE(&(e)->dfi) >= sizeof(grn_expr_dfi)) {\
|
|
GRN_BULK_INCR_LEN((&(e)->dfi), -(sizeof(grn_expr_dfi)));\
|
|
(d) = (grn_expr_dfi *)(GRN_BULK_CURR(&(e)->dfi));\
|
|
(e)->code0 = (d)->code;\
|
|
} else {\
|
|
(d) = NULL;\
|
|
(e)->code0 = NULL;\
|
|
}\
|
|
} while (0)
|
|
|
|
#define DFI_PUT(e,t,d,c) do {\
|
|
grn_expr_dfi dfi;\
|
|
dfi.type = (t);\
|
|
dfi.domain = (d);\
|
|
dfi.code = (c);\
|
|
if ((e)->code0) { (e)->code0->modify = (c) ? ((c) - (e)->code0) : 0; }\
|
|
grn_bulk_write(ctx, &(e)->dfi, (char *)&dfi, sizeof(grn_expr_dfi));\
|
|
(e)->code0 = NULL;\
|
|
} while (0)
|
|
|
|
grn_expr_dfi *
|
|
dfi_value_at(grn_expr *expr, int offset)
|
|
{
|
|
grn_obj *obj = &expr->dfi;
|
|
int size = GRN_BULK_VSIZE(obj) / sizeof(grn_expr_dfi);
|
|
if (offset < 0) { offset = size + offset; }
|
|
return (0 <= offset && offset < size)
|
|
? &(((grn_expr_dfi *)GRN_BULK_HEAD(obj))[offset])
|
|
: NULL;
|
|
}
|
|
|
|
grn_obj *
|
|
grn_expr_create(grn_ctx *ctx, const char *name, unsigned int name_size)
|
|
{
|
|
grn_id id;
|
|
grn_obj *db;
|
|
grn_expr *expr = NULL;
|
|
if (!ctx || !ctx->impl || !(db = ctx->impl->db)) {
|
|
ERR(GRN_INVALID_ARGUMENT, "db not initialized");
|
|
return NULL;
|
|
}
|
|
if (name_size) {
|
|
ERR(GRN_FUNCTION_NOT_IMPLEMENTED,
|
|
"[expr][create] anonymous expression isn't implemented yet");
|
|
return NULL;
|
|
}
|
|
GRN_API_ENTER;
|
|
if (grn_db_check_name(ctx, name, name_size)) {
|
|
GRN_DB_CHECK_NAME_ERR("[expr][create]", name, name_size);
|
|
GRN_API_RETURN(NULL);
|
|
}
|
|
if (!GRN_DB_P(db)) {
|
|
ERR(GRN_INVALID_ARGUMENT, "named expr is not supported");
|
|
GRN_API_RETURN(NULL);
|
|
}
|
|
id = grn_obj_register(ctx, db, name, name_size);
|
|
if (id && (expr = GRN_MALLOCN(grn_expr, 1))) {
|
|
int size = GRN_STACK_SIZE;
|
|
expr->consts = NULL;
|
|
expr->nconsts = 0;
|
|
GRN_TEXT_INIT(&expr->name_buf, 0);
|
|
GRN_TEXT_INIT(&expr->dfi, 0);
|
|
GRN_PTR_INIT(&expr->objs, GRN_OBJ_VECTOR, GRN_ID_NIL);
|
|
expr->code0 = NULL;
|
|
expr->vars = NULL;
|
|
expr->nvars = 0;
|
|
expr->cacheable = 1;
|
|
expr->taintable = 0;
|
|
expr->values_curr = 0;
|
|
expr->values_tail = 0;
|
|
expr->values_size = size;
|
|
expr->codes_curr = 0;
|
|
expr->codes_size = size;
|
|
GRN_DB_OBJ_SET_TYPE(expr, GRN_EXPR);
|
|
expr->obj.header.domain = GRN_ID_NIL;
|
|
expr->obj.range = GRN_ID_NIL;
|
|
if (!grn_db_obj_init(ctx, db, id, DB_OBJ(expr))) {
|
|
if ((expr->values = GRN_MALLOCN(grn_obj, size))) {
|
|
int i;
|
|
for (i = 0; i < size; i++) {
|
|
GRN_OBJ_INIT(&expr->values[i], GRN_BULK, GRN_OBJ_EXPRVALUE, GRN_ID_NIL);
|
|
}
|
|
if ((expr->codes = GRN_MALLOCN(grn_expr_code, size))) {
|
|
goto exit;
|
|
}
|
|
GRN_FREE(expr->values);
|
|
}
|
|
}
|
|
GRN_FREE(expr);
|
|
expr = NULL;
|
|
}
|
|
exit :
|
|
GRN_API_RETURN((grn_obj *)expr);
|
|
}
|
|
|
|
#define GRN_PTR_POP(obj,value) do {\
|
|
if (GRN_BULK_VSIZE(obj) >= sizeof(grn_obj *)) {\
|
|
GRN_BULK_INCR_LEN((obj), -(sizeof(grn_obj *)));\
|
|
value = *(grn_obj **)(GRN_BULK_CURR(obj));\
|
|
} else {\
|
|
value = NULL;\
|
|
}\
|
|
} while (0)
|
|
|
|
grn_rc
|
|
grn_expr_close(grn_ctx *ctx, grn_obj *expr)
|
|
{
|
|
uint32_t i;
|
|
grn_expr *e = (grn_expr *)expr;
|
|
GRN_API_ENTER;
|
|
/*
|
|
if (e->obj.header.domain) {
|
|
grn_hash_delete(ctx, ctx->impl->qe, &e->obj.header.domain, sizeof(grn_id), NULL);
|
|
}
|
|
*/
|
|
grn_expr_clear_vars(ctx, expr);
|
|
for (i = 0; i < e->nconsts; i++) {
|
|
grn_obj_close(ctx, &e->consts[i]);
|
|
}
|
|
if (e->consts) { GRN_FREE(e->consts); }
|
|
grn_obj_close(ctx, &e->name_buf);
|
|
grn_obj_close(ctx, &e->dfi);
|
|
for (;;) {
|
|
grn_obj *obj;
|
|
GRN_PTR_POP(&e->objs, obj);
|
|
if (obj) {
|
|
#ifdef USE_MEMORY_DEBUG
|
|
grn_obj_unlink(ctx, obj);
|
|
#else
|
|
if (obj->header.type) {
|
|
grn_obj_unlink(ctx, obj);
|
|
} else {
|
|
GRN_LOG(ctx, GRN_LOG_WARNING, "GRN_VOID object is tried to be unlinked");
|
|
}
|
|
#endif
|
|
} else { break; }
|
|
}
|
|
grn_obj_close(ctx, &e->objs);
|
|
for (i = 0; i < e->nvars; i++) {
|
|
grn_obj_close(ctx, &e->vars[i].value);
|
|
}
|
|
if (e->vars) { GRN_FREE(e->vars); }
|
|
for (i = 0; i < e->values_tail; i++) {
|
|
grn_obj_close(ctx, &e->values[i]);
|
|
}
|
|
GRN_FREE(e->values);
|
|
GRN_FREE(e->codes);
|
|
GRN_FREE(e);
|
|
GRN_API_RETURN(ctx->rc);
|
|
}
|
|
|
|
grn_obj *
|
|
grn_expr_add_var(grn_ctx *ctx, grn_obj *expr, const char *name, unsigned int name_size)
|
|
{
|
|
uint32_t i;
|
|
char *p;
|
|
grn_expr_var *v;
|
|
grn_obj *res = NULL;
|
|
grn_expr *e = (grn_expr *)expr;
|
|
GRN_API_ENTER;
|
|
if (DB_OBJ(expr)->id & GRN_OBJ_TMP_OBJECT) {
|
|
res = grn_expr_get_or_add_var(ctx, expr, name, name_size);
|
|
} else {
|
|
if (!e->vars) {
|
|
if (!(e->vars = GRN_MALLOCN(grn_expr_var, GRN_STACK_SIZE))) {
|
|
ERR(GRN_NO_MEMORY_AVAILABLE, "malloc failed");
|
|
}
|
|
}
|
|
if (e->vars && e->nvars < GRN_STACK_SIZE) {
|
|
v = e->vars + e->nvars++;
|
|
if (name_size) {
|
|
GRN_TEXT_PUT(ctx, &e->name_buf, name, name_size);
|
|
} else {
|
|
uint32_t ol = GRN_TEXT_LEN(&e->name_buf);
|
|
GRN_TEXT_PUTC(ctx, &e->name_buf, '$');
|
|
grn_text_itoa(ctx, &e->name_buf, e->nvars);
|
|
name_size = GRN_TEXT_LEN(&e->name_buf) - ol;
|
|
}
|
|
v->name_size = name_size;
|
|
res = &v->value;
|
|
GRN_VOID_INIT(res);
|
|
for (i = e->nvars, p = GRN_TEXT_VALUE(&e->name_buf), v = e->vars; i; i--, v++) {
|
|
v->name = p;
|
|
p += v->name_size;
|
|
}
|
|
}
|
|
}
|
|
GRN_API_RETURN(res);
|
|
}
|
|
|
|
grn_obj *
|
|
grn_expr_get_var(grn_ctx *ctx, grn_obj *expr, const char *name, unsigned int name_size)
|
|
{
|
|
uint32_t n;
|
|
grn_obj *res = NULL;
|
|
grn_hash *vars = grn_expr_get_vars(ctx, expr, &n);
|
|
if (vars) { grn_hash_get(ctx, vars, name, name_size, (void **)&res); }
|
|
return res;
|
|
}
|
|
|
|
grn_obj *
|
|
grn_expr_get_or_add_var(grn_ctx *ctx, grn_obj *expr, const char *name, unsigned int name_size)
|
|
{
|
|
uint32_t n;
|
|
grn_obj *res = NULL;
|
|
grn_hash *vars = grn_expr_get_vars(ctx, expr, &n);
|
|
if (vars) {
|
|
int added = 0;
|
|
char name_buf[16];
|
|
if (!name_size) {
|
|
char *rest;
|
|
name_buf[0] = '$';
|
|
grn_itoa((int)GRN_HASH_SIZE(vars) + 1, name_buf + 1, name_buf + 16, &rest);
|
|
name_size = rest - name_buf;
|
|
name = name_buf;
|
|
}
|
|
grn_hash_add(ctx, vars, name, name_size, (void **)&res, &added);
|
|
if (added) { GRN_TEXT_INIT(res, 0); }
|
|
}
|
|
return res;
|
|
}
|
|
|
|
grn_obj *
|
|
grn_expr_get_var_by_offset(grn_ctx *ctx, grn_obj *expr, unsigned int offset)
|
|
{
|
|
uint32_t n;
|
|
grn_obj *res = NULL;
|
|
grn_hash *vars = grn_expr_get_vars(ctx, expr, &n);
|
|
if (vars) { res = (grn_obj *)grn_hash_get_value_(ctx, vars, offset + 1, &n); }
|
|
return res;
|
|
}
|
|
|
|
#define EXPRVP(x) ((x)->header.impl_flags & GRN_OBJ_EXPRVALUE)
|
|
|
|
#define CONSTP(obj) ((obj) && ((obj)->header.impl_flags & GRN_OBJ_EXPRCONST))
|
|
|
|
#define PUSH_CODE(e,o,v,n,c) do {\
|
|
(c) = &(e)->codes[e->codes_curr++];\
|
|
(c)->value = (v);\
|
|
(c)->nargs = (n);\
|
|
(c)->op = (o);\
|
|
(c)->flags = 0;\
|
|
(c)->modify = 0;\
|
|
} while (0)
|
|
|
|
#define APPEND_UNARY_MINUS_OP(e) do { \
|
|
grn_expr_code *code_; \
|
|
grn_id domain; \
|
|
unsigned char type; \
|
|
grn_obj *x; \
|
|
DFI_POP(e, dfi); \
|
|
code_ = dfi->code; \
|
|
domain = dfi->domain; \
|
|
type = dfi->type; \
|
|
x = code_->value; \
|
|
if (CONSTP(x)) { \
|
|
switch (domain) { \
|
|
case GRN_DB_INT32: \
|
|
{ \
|
|
int value; \
|
|
value = GRN_INT32_VALUE(x); \
|
|
if (value == (int)0x80000000) { \
|
|
domain = GRN_DB_INT64; \
|
|
x->header.domain = domain; \
|
|
GRN_INT64_SET(ctx, x, -((long long int)value)); \
|
|
} else { \
|
|
GRN_INT32_SET(ctx, x, -value); \
|
|
} \
|
|
} \
|
|
break; \
|
|
case GRN_DB_UINT32: \
|
|
{ \
|
|
unsigned int value; \
|
|
value = GRN_UINT32_VALUE(x); \
|
|
if (value > (unsigned int)0x80000000) { \
|
|
domain = GRN_DB_INT64; \
|
|
x->header.domain = domain; \
|
|
GRN_INT64_SET(ctx, x, -((long long int)value)); \
|
|
} else { \
|
|
domain = GRN_DB_INT32; \
|
|
x->header.domain = domain; \
|
|
GRN_INT32_SET(ctx, x, -((int)value)); \
|
|
} \
|
|
} \
|
|
break; \
|
|
case GRN_DB_INT64: \
|
|
GRN_INT64_SET(ctx, x, -GRN_INT64_VALUE(x)); \
|
|
break; \
|
|
case GRN_DB_FLOAT: \
|
|
GRN_FLOAT_SET(ctx, x, -GRN_FLOAT_VALUE(x)); \
|
|
break; \
|
|
default: \
|
|
PUSH_CODE(e, op, obj, nargs, code); \
|
|
break; \
|
|
} \
|
|
} else { \
|
|
PUSH_CODE(e, op, obj, nargs, code); \
|
|
} \
|
|
DFI_PUT(e, type, domain, code_); \
|
|
} while (0)
|
|
|
|
#define PUSH_N_ARGS_ARITHMETIC_OP(e, op, obj, nargs, code) do { \
|
|
PUSH_CODE(e, op, obj, nargs, code); \
|
|
{ \
|
|
int i = nargs; \
|
|
while (i--) { \
|
|
DFI_POP(e, dfi); \
|
|
} \
|
|
} \
|
|
DFI_PUT(e, type, domain, code); \
|
|
} while (0)
|
|
|
|
grn_obj *
|
|
grn_expr_append_obj(grn_ctx *ctx, grn_obj *expr, grn_obj *obj, grn_operator op, int nargs)
|
|
{
|
|
uint8_t type = GRN_VOID;
|
|
grn_id domain = GRN_ID_NIL;
|
|
grn_expr_dfi *dfi;
|
|
grn_expr_code *code;
|
|
grn_obj *res = NULL;
|
|
grn_expr *e = (grn_expr *)expr;
|
|
GRN_API_ENTER;
|
|
if (e->codes_curr >= e->codes_size) {
|
|
ERR(GRN_NO_MEMORY_AVAILABLE, "stack is full");
|
|
goto exit;
|
|
}
|
|
{
|
|
switch (op) {
|
|
case GRN_OP_PUSH :
|
|
if (obj) {
|
|
PUSH_CODE(e, op, obj, nargs, code);
|
|
DFI_PUT(e, obj->header.type, GRN_OBJ_GET_DOMAIN(obj), code);
|
|
} else {
|
|
ERR(GRN_INVALID_ARGUMENT, "obj not assigned for GRN_OP_PUSH");
|
|
goto exit;
|
|
}
|
|
break;
|
|
case GRN_OP_NOP :
|
|
/* nop */
|
|
break;
|
|
case GRN_OP_POP :
|
|
if (obj) {
|
|
ERR(GRN_INVALID_ARGUMENT, "obj assigned for GRN_OP_POP");
|
|
goto exit;
|
|
} else {
|
|
PUSH_CODE(e, op, obj, nargs, code);
|
|
DFI_POP(e, dfi);
|
|
}
|
|
break;
|
|
case GRN_OP_CALL :
|
|
{
|
|
grn_obj *proc = NULL;
|
|
if (e->codes_curr - nargs > 0) {
|
|
int i;
|
|
grn_expr_code *code;
|
|
code = &(e->codes[e->codes_curr - 1]);
|
|
for (i = 0; i < nargs; i++) {
|
|
int rest_n_codes = 1;
|
|
while (rest_n_codes > 0) {
|
|
if (!code->value) {
|
|
rest_n_codes += code->nargs;
|
|
}
|
|
rest_n_codes--;
|
|
code--;
|
|
}
|
|
}
|
|
proc = code->value;
|
|
}
|
|
if (!proc) {
|
|
ERR(GRN_INVALID_ARGUMENT, "invalid function call expression");
|
|
goto exit;
|
|
}
|
|
if (!(grn_obj_is_function_proc(ctx, proc) ||
|
|
grn_obj_is_scorer_proc(ctx, proc))) {
|
|
grn_obj buffer;
|
|
|
|
GRN_TEXT_INIT(&buffer, 0);
|
|
switch (proc->header.type) {
|
|
case GRN_TABLE_HASH_KEY:
|
|
case GRN_TABLE_PAT_KEY:
|
|
case GRN_TABLE_NO_KEY:
|
|
case GRN_COLUMN_FIX_SIZE:
|
|
case GRN_COLUMN_VAR_SIZE:
|
|
case GRN_COLUMN_INDEX:
|
|
grn_inspect_name(ctx, &buffer, proc);
|
|
break;
|
|
default:
|
|
grn_inspect(ctx, &buffer, proc);
|
|
break;
|
|
}
|
|
ERR(GRN_INVALID_ARGUMENT, "invalid function: <%.*s>",
|
|
(int)GRN_TEXT_LEN(&buffer), GRN_TEXT_VALUE(&buffer));
|
|
GRN_OBJ_FIN(ctx, &buffer);
|
|
goto exit;
|
|
}
|
|
}
|
|
PUSH_CODE(e, op, obj, nargs, code);
|
|
{
|
|
int i = nargs;
|
|
while (i--) { DFI_POP(e, dfi); }
|
|
}
|
|
if (!obj) { DFI_POP(e, dfi); }
|
|
// todo : increment e->values_tail.
|
|
DFI_PUT(e, type, domain, code); /* cannot identify type of return value */
|
|
e->cacheable = 0;
|
|
break;
|
|
case GRN_OP_INTERN :
|
|
if (obj && CONSTP(obj)) {
|
|
grn_obj *value;
|
|
value = grn_expr_get_var(ctx, expr, GRN_TEXT_VALUE(obj), GRN_TEXT_LEN(obj));
|
|
if (!value) { value = grn_ctx_get(ctx, GRN_TEXT_VALUE(obj), GRN_TEXT_LEN(obj)); }
|
|
if (value) {
|
|
obj = value;
|
|
op = GRN_OP_PUSH;
|
|
type = obj->header.type;
|
|
domain = GRN_OBJ_GET_DOMAIN(obj);
|
|
}
|
|
}
|
|
PUSH_CODE(e, op, obj, nargs, code);
|
|
DFI_PUT(e, type, domain, code);
|
|
break;
|
|
case GRN_OP_EQUAL :
|
|
PUSH_CODE(e, op, obj, nargs, code);
|
|
if (nargs) {
|
|
grn_id xd, yd = GRN_ID_NIL;
|
|
grn_obj *x, *y = NULL;
|
|
int i = nargs - 1;
|
|
if (obj) {
|
|
xd = GRN_OBJ_GET_DOMAIN(obj);
|
|
x = obj;
|
|
} else {
|
|
DFI_POP(e, dfi);
|
|
x = dfi->code->value;
|
|
xd = dfi->domain;
|
|
}
|
|
while (i--) {
|
|
DFI_POP(e, dfi);
|
|
y = dfi->code->value;
|
|
yd = dfi->domain;
|
|
}
|
|
if (CONSTP(x)) {
|
|
if (CONSTP(y)) {
|
|
/* todo */
|
|
} else {
|
|
grn_obj dest;
|
|
if (xd != yd) {
|
|
GRN_OBJ_INIT(&dest, GRN_BULK, 0, yd);
|
|
if (!grn_obj_cast(ctx, x, &dest, GRN_FALSE)) {
|
|
grn_obj_reinit(ctx, x, yd, 0);
|
|
grn_bulk_write(ctx, x, GRN_BULK_HEAD(&dest), GRN_BULK_VSIZE(&dest));
|
|
}
|
|
GRN_OBJ_FIN(ctx, &dest);
|
|
}
|
|
}
|
|
} else {
|
|
if (CONSTP(y)) {
|
|
grn_obj dest;
|
|
if (xd != yd) {
|
|
GRN_OBJ_INIT(&dest, GRN_BULK, 0, xd);
|
|
if (!grn_obj_cast(ctx, y, &dest, GRN_FALSE)) {
|
|
grn_obj_reinit(ctx, y, xd, 0);
|
|
grn_bulk_write(ctx, y, GRN_BULK_HEAD(&dest), GRN_BULK_VSIZE(&dest));
|
|
}
|
|
GRN_OBJ_FIN(ctx, &dest);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
DFI_PUT(e, type, domain, code);
|
|
break;
|
|
case GRN_OP_TABLE_CREATE :
|
|
case GRN_OP_EXPR_GET_VAR :
|
|
case GRN_OP_MATCH :
|
|
case GRN_OP_NEAR :
|
|
case GRN_OP_NEAR2 :
|
|
case GRN_OP_SIMILAR :
|
|
case GRN_OP_PREFIX :
|
|
case GRN_OP_SUFFIX :
|
|
case GRN_OP_NOT_EQUAL :
|
|
case GRN_OP_LESS :
|
|
case GRN_OP_GREATER :
|
|
case GRN_OP_LESS_EQUAL :
|
|
case GRN_OP_GREATER_EQUAL :
|
|
case GRN_OP_GEO_DISTANCE1 :
|
|
case GRN_OP_GEO_DISTANCE2 :
|
|
case GRN_OP_GEO_DISTANCE3 :
|
|
case GRN_OP_GEO_DISTANCE4 :
|
|
case GRN_OP_GEO_WITHINP5 :
|
|
case GRN_OP_GEO_WITHINP6 :
|
|
case GRN_OP_GEO_WITHINP8 :
|
|
case GRN_OP_OBJ_SEARCH :
|
|
case GRN_OP_TABLE_SELECT :
|
|
case GRN_OP_TABLE_SORT :
|
|
case GRN_OP_TABLE_GROUP :
|
|
case GRN_OP_JSON_PUT :
|
|
case GRN_OP_GET_REF :
|
|
case GRN_OP_ADJUST :
|
|
case GRN_OP_TERM_EXTRACT :
|
|
PUSH_CODE(e, op, obj, nargs, code);
|
|
if (nargs) {
|
|
int i = nargs - 1;
|
|
if (!obj) { DFI_POP(e, dfi); }
|
|
while (i--) { DFI_POP(e, dfi); }
|
|
}
|
|
DFI_PUT(e, type, domain, code);
|
|
break;
|
|
case GRN_OP_AND :
|
|
case GRN_OP_OR :
|
|
case GRN_OP_AND_NOT :
|
|
PUSH_CODE(e, op, obj, nargs, code);
|
|
if (nargs != 2) {
|
|
GRN_LOG(ctx, GRN_LOG_WARNING, "nargs(%d) != 2 in relative op", nargs);
|
|
}
|
|
if (obj) {
|
|
GRN_LOG(ctx, GRN_LOG_WARNING, "obj assigned to relative op");
|
|
}
|
|
{
|
|
int i = nargs;
|
|
while (i--) {
|
|
DFI_POP(e, dfi);
|
|
if (dfi) {
|
|
dfi->code->flags |= GRN_EXPR_CODE_RELATIONAL_EXPRESSION;
|
|
} else {
|
|
ERR(GRN_SYNTAX_ERROR, "stack under flow in relative op");
|
|
}
|
|
}
|
|
}
|
|
DFI_PUT(e, type, domain, code);
|
|
break;
|
|
case GRN_OP_NOT :
|
|
if (nargs == 1) {
|
|
PUSH_CODE(e, op, obj, nargs, code);
|
|
}
|
|
break;
|
|
case GRN_OP_PLUS :
|
|
if (nargs > 1) {
|
|
PUSH_N_ARGS_ARITHMETIC_OP(e, op, obj, nargs, code);
|
|
}
|
|
break;
|
|
case GRN_OP_MINUS :
|
|
if (nargs == 1) {
|
|
APPEND_UNARY_MINUS_OP(e);
|
|
} else {
|
|
PUSH_N_ARGS_ARITHMETIC_OP(e, op, obj, nargs, code);
|
|
}
|
|
break;
|
|
case GRN_OP_BITWISE_NOT :
|
|
PUSH_CODE(e, op, obj, nargs, code);
|
|
break;
|
|
case GRN_OP_STAR :
|
|
case GRN_OP_SLASH :
|
|
case GRN_OP_MOD :
|
|
case GRN_OP_SHIFTL :
|
|
case GRN_OP_SHIFTR :
|
|
case GRN_OP_SHIFTRR :
|
|
case GRN_OP_BITWISE_OR :
|
|
case GRN_OP_BITWISE_XOR :
|
|
case GRN_OP_BITWISE_AND :
|
|
PUSH_N_ARGS_ARITHMETIC_OP(e, op, obj, nargs, code);
|
|
break;
|
|
case GRN_OP_INCR :
|
|
case GRN_OP_DECR :
|
|
case GRN_OP_INCR_POST :
|
|
case GRN_OP_DECR_POST :
|
|
{
|
|
DFI_POP(e, dfi);
|
|
if (dfi) {
|
|
type = dfi->type;
|
|
domain = dfi->domain;
|
|
if (dfi->code) {
|
|
if (dfi->code->op == GRN_OP_GET_VALUE) {
|
|
dfi->code->op = GRN_OP_GET_REF;
|
|
}
|
|
if (dfi->code->value && grn_obj_is_persistent(ctx, dfi->code->value)) {
|
|
e->cacheable = 0;
|
|
e->taintable = 1;
|
|
}
|
|
}
|
|
}
|
|
PUSH_CODE(e, op, obj, nargs, code);
|
|
}
|
|
DFI_PUT(e, type, domain, code);
|
|
break;
|
|
case GRN_OP_GET_VALUE :
|
|
{
|
|
grn_id vdomain = GRN_ID_NIL;
|
|
if (obj) {
|
|
if (nargs == 1) {
|
|
grn_obj *v = grn_expr_get_var_by_offset(ctx, expr, 0);
|
|
if (v) { vdomain = GRN_OBJ_GET_DOMAIN(v); }
|
|
} else {
|
|
DFI_POP(e, dfi);
|
|
vdomain = dfi->domain;
|
|
}
|
|
if (vdomain && CONSTP(obj) && obj->header.type == GRN_BULK) {
|
|
grn_obj *table = grn_ctx_at(ctx, vdomain);
|
|
grn_obj *col = grn_obj_column(ctx, table, GRN_BULK_HEAD(obj), GRN_BULK_VSIZE(obj));
|
|
if (col) {
|
|
obj = col;
|
|
type = col->header.type;
|
|
domain = grn_obj_get_range(ctx, col);
|
|
grn_expr_take_obj(ctx, (grn_obj *)e, col);
|
|
}
|
|
} else {
|
|
domain = grn_obj_get_range(ctx, obj);
|
|
}
|
|
PUSH_CODE(e, op, obj, nargs, code);
|
|
} else {
|
|
grn_expr_dfi *dfi0;
|
|
DFI_POP(e, dfi0);
|
|
if (nargs == 1) {
|
|
grn_obj *v = grn_expr_get_var_by_offset(ctx, expr, 0);
|
|
if (v) { vdomain = GRN_OBJ_GET_DOMAIN(v); }
|
|
} else {
|
|
DFI_POP(e, dfi);
|
|
vdomain = dfi->domain;
|
|
}
|
|
if (dfi0->code->op == GRN_OP_PUSH) {
|
|
dfi0->code->op = op;
|
|
dfi0->code->nargs = nargs;
|
|
obj = dfi0->code->value;
|
|
if (vdomain && obj && CONSTP(obj) && obj->header.type == GRN_BULK) {
|
|
grn_obj *table = grn_ctx_at(ctx, vdomain);
|
|
grn_obj *col = grn_obj_column(ctx, table, GRN_BULK_HEAD(obj), GRN_BULK_VSIZE(obj));
|
|
if (col) {
|
|
dfi0->code->value = col;
|
|
type = col->header.type;
|
|
domain = grn_obj_get_range(ctx, col);
|
|
grn_obj_unlink(ctx, col);
|
|
}
|
|
} else {
|
|
domain = grn_obj_get_range(ctx, obj);
|
|
}
|
|
code = dfi0->code;
|
|
} else {
|
|
PUSH_CODE(e, op, obj, nargs, code);
|
|
}
|
|
}
|
|
}
|
|
DFI_PUT(e, type, domain, code);
|
|
break;
|
|
case GRN_OP_ASSIGN :
|
|
case GRN_OP_STAR_ASSIGN :
|
|
case GRN_OP_SLASH_ASSIGN :
|
|
case GRN_OP_MOD_ASSIGN :
|
|
case GRN_OP_PLUS_ASSIGN :
|
|
case GRN_OP_MINUS_ASSIGN :
|
|
case GRN_OP_SHIFTL_ASSIGN :
|
|
case GRN_OP_SHIFTR_ASSIGN :
|
|
case GRN_OP_SHIFTRR_ASSIGN :
|
|
case GRN_OP_AND_ASSIGN :
|
|
case GRN_OP_OR_ASSIGN :
|
|
case GRN_OP_XOR_ASSIGN :
|
|
{
|
|
if (obj) {
|
|
type = obj->header.type;
|
|
domain = GRN_OBJ_GET_DOMAIN(obj);
|
|
} else {
|
|
DFI_POP(e, dfi);
|
|
if (dfi) {
|
|
type = dfi->type;
|
|
domain = dfi->domain;
|
|
}
|
|
}
|
|
DFI_POP(e, dfi);
|
|
if (dfi && (dfi->code)) {
|
|
if (dfi->code->op == GRN_OP_GET_VALUE) {
|
|
dfi->code->op = GRN_OP_GET_REF;
|
|
}
|
|
if (dfi->code->value && grn_obj_is_persistent(ctx, dfi->code->value)) {
|
|
e->cacheable = 0;
|
|
e->taintable = 1;
|
|
}
|
|
}
|
|
PUSH_CODE(e, op, obj, nargs, code);
|
|
}
|
|
DFI_PUT(e, type, domain, code);
|
|
break;
|
|
case GRN_OP_JUMP :
|
|
DFI_POP(e, dfi);
|
|
PUSH_CODE(e, op, obj, nargs, code);
|
|
break;
|
|
case GRN_OP_CJUMP :
|
|
DFI_POP(e, dfi);
|
|
PUSH_CODE(e, op, obj, nargs, code);
|
|
break;
|
|
case GRN_OP_COMMA :
|
|
PUSH_CODE(e, op, obj, nargs, code);
|
|
break;
|
|
case GRN_OP_GET_MEMBER :
|
|
DFI_POP(e, dfi);
|
|
DFI_POP(e, dfi);
|
|
if (dfi) {
|
|
type = dfi->type;
|
|
domain = dfi->domain;
|
|
if (dfi->code) {
|
|
if (dfi->code->op == GRN_OP_GET_VALUE) {
|
|
dfi->code->op = GRN_OP_GET_REF;
|
|
}
|
|
}
|
|
}
|
|
PUSH_CODE(e, op, obj, nargs, code);
|
|
DFI_PUT(e, type, domain, code);
|
|
break;
|
|
default :
|
|
break;
|
|
}
|
|
}
|
|
exit :
|
|
if (!ctx->rc) { res = obj; }
|
|
GRN_API_RETURN(res);
|
|
}
|
|
#undef PUSH_N_ARGS_ARITHMETIC_OP
|
|
#undef APPEND_UNARY_MINUS_OP
|
|
|
|
grn_obj *
|
|
grn_expr_append_const(grn_ctx *ctx, grn_obj *expr, grn_obj *obj,
|
|
grn_operator op, int nargs)
|
|
{
|
|
grn_obj *res = NULL;
|
|
GRN_API_ENTER;
|
|
if (!obj) {
|
|
ERR(GRN_SYNTAX_ERROR, "constant is null");
|
|
goto exit;
|
|
}
|
|
if (GRN_DB_OBJP(obj) || GRN_ACCESSORP(obj)) {
|
|
res = obj;
|
|
} else {
|
|
if ((res = grn_expr_alloc_const(ctx, expr))) {
|
|
switch (obj->header.type) {
|
|
case GRN_VOID :
|
|
case GRN_BULK :
|
|
case GRN_UVECTOR :
|
|
GRN_OBJ_INIT(res, obj->header.type, 0, obj->header.domain);
|
|
grn_bulk_write(ctx, res, GRN_BULK_HEAD(obj), GRN_BULK_VSIZE(obj));
|
|
break;
|
|
default :
|
|
res = NULL;
|
|
ERR(GRN_FUNCTION_NOT_IMPLEMENTED, "unsupported type");
|
|
goto exit;
|
|
}
|
|
res->header.impl_flags |= GRN_OBJ_EXPRCONST;
|
|
}
|
|
}
|
|
grn_expr_append_obj(ctx, expr, res, op, nargs); /* constant */
|
|
exit :
|
|
GRN_API_RETURN(res);
|
|
}
|
|
|
|
static grn_obj *
|
|
grn_expr_add_str(grn_ctx *ctx, grn_obj *expr, const char *str, unsigned int str_size)
|
|
{
|
|
grn_obj *res = NULL;
|
|
if ((res = grn_expr_alloc_const(ctx, expr))) {
|
|
GRN_TEXT_INIT(res, 0);
|
|
grn_bulk_write(ctx, res, str, str_size);
|
|
res->header.impl_flags |= GRN_OBJ_EXPRCONST;
|
|
}
|
|
return res;
|
|
}
|
|
|
|
grn_obj *
|
|
grn_expr_append_const_str(grn_ctx *ctx, grn_obj *expr, const char *str, unsigned int str_size,
|
|
grn_operator op, int nargs)
|
|
{
|
|
grn_obj *res;
|
|
GRN_API_ENTER;
|
|
res = grn_expr_add_str(ctx, expr, str, str_size);
|
|
grn_expr_append_obj(ctx, expr, res, op, nargs); /* constant */
|
|
GRN_API_RETURN(res);
|
|
}
|
|
|
|
grn_obj *
|
|
grn_expr_append_const_int(grn_ctx *ctx, grn_obj *expr, int i,
|
|
grn_operator op, int nargs)
|
|
{
|
|
grn_obj *res = NULL;
|
|
GRN_API_ENTER;
|
|
if ((res = grn_expr_alloc_const(ctx, expr))) {
|
|
GRN_INT32_INIT(res, 0);
|
|
GRN_INT32_SET(ctx, res, i);
|
|
res->header.impl_flags |= GRN_OBJ_EXPRCONST;
|
|
}
|
|
grn_expr_append_obj(ctx, expr, res, op, nargs); /* constant */
|
|
GRN_API_RETURN(res);
|
|
}
|
|
|
|
grn_rc
|
|
grn_expr_append_op(grn_ctx *ctx, grn_obj *expr, grn_operator op, int nargs)
|
|
{
|
|
grn_expr_append_obj(ctx, expr, NULL, op, nargs);
|
|
return ctx->rc;
|
|
}
|
|
|
|
grn_rc
|
|
grn_expr_compile(grn_ctx *ctx, grn_obj *expr)
|
|
{
|
|
grn_obj_spec_save(ctx, DB_OBJ(expr));
|
|
return ctx->rc;
|
|
}
|
|
|
|
#define WITH_SPSAVE(block) do {\
|
|
ctx->impl->stack_curr = sp - ctx->impl->stack;\
|
|
e->values_curr = vp - e->values;\
|
|
block\
|
|
vp = e->values + e->values_curr;\
|
|
sp = ctx->impl->stack + ctx->impl->stack_curr;\
|
|
s0 = sp[-1];\
|
|
s1 = sp[-2];\
|
|
} while (0)
|
|
|
|
#define GEO_RESOLUTION 3600000
|
|
#define GEO_RADIOUS 6357303
|
|
#define GEO_BES_C1 6334834
|
|
#define GEO_BES_C2 6377397
|
|
#define GEO_BES_C3 0.006674
|
|
#define GEO_GRS_C1 6335439
|
|
#define GEO_GRS_C2 6378137
|
|
#define GEO_GRS_C3 0.006694
|
|
#define GEO_INT2RAD(x) ((M_PI * x) / (GEO_RESOLUTION * 180))
|
|
|
|
#define VAR_SET_VALUE(ctx,var,value) do {\
|
|
if (GRN_DB_OBJP(value)) {\
|
|
(var)->header.type = GRN_PTR;\
|
|
(var)->header.domain = DB_OBJ(value)->id;\
|
|
GRN_PTR_SET(ctx, (var), (value));\
|
|
} else {\
|
|
(var)->header.type = (value)->header.type;\
|
|
(var)->header.domain = (value)->header.domain;\
|
|
GRN_TEXT_SET(ctx, (var), GRN_TEXT_VALUE(value), GRN_TEXT_LEN(value));\
|
|
}\
|
|
} while (0)
|
|
|
|
grn_rc
|
|
grn_proc_call(grn_ctx *ctx, grn_obj *proc, int nargs, grn_obj *caller)
|
|
{
|
|
grn_proc_ctx pctx;
|
|
grn_obj *obj = NULL, **args;
|
|
grn_proc *p = (grn_proc *)proc;
|
|
if (nargs > ctx->impl->stack_curr) { return GRN_INVALID_ARGUMENT; }
|
|
GRN_API_ENTER;
|
|
args = ctx->impl->stack + ctx->impl->stack_curr - nargs;
|
|
pctx.proc = p;
|
|
pctx.caller = caller;
|
|
pctx.user_data.ptr = NULL;
|
|
if (p->funcs[PROC_INIT]) {
|
|
obj = p->funcs[PROC_INIT](ctx, nargs, args, &pctx.user_data);
|
|
}
|
|
pctx.phase = PROC_NEXT;
|
|
if (p->funcs[PROC_NEXT]) {
|
|
obj = p->funcs[PROC_NEXT](ctx, nargs, args, &pctx.user_data);
|
|
}
|
|
pctx.phase = PROC_FIN;
|
|
if (p->funcs[PROC_FIN]) {
|
|
obj = p->funcs[PROC_FIN](ctx, nargs, args, &pctx.user_data);
|
|
}
|
|
ctx->impl->stack_curr -= nargs;
|
|
grn_ctx_push(ctx, obj);
|
|
GRN_API_RETURN(ctx->rc);
|
|
}
|
|
|
|
#define PUSH1(v) do {\
|
|
if (EXPRVP(v)) { vp++; }\
|
|
s1 = s0;\
|
|
*sp++ = s0 = v;\
|
|
} while (0)
|
|
|
|
#define POP1(v) do {\
|
|
if (EXPRVP(s0)) { vp--; }\
|
|
v = s0;\
|
|
s0 = s1;\
|
|
sp--;\
|
|
if (sp < s_) { ERR(GRN_INVALID_ARGUMENT, "stack underflow"); goto exit; }\
|
|
s1 = sp[-2];\
|
|
} while (0)
|
|
|
|
#define ALLOC1(value) do {\
|
|
s1 = s0;\
|
|
*sp++ = s0 = value = vp++;\
|
|
if (vp - e->values > e->values_tail) { e->values_tail = vp - e->values; }\
|
|
} while (0)
|
|
|
|
#define POP1ALLOC1(arg,value) do {\
|
|
arg = s0;\
|
|
if (EXPRVP(s0)) {\
|
|
value = s0;\
|
|
} else {\
|
|
if (sp < s_ + 1) { ERR(GRN_INVALID_ARGUMENT, "stack underflow"); goto exit; }\
|
|
sp[-1] = s0 = value = vp++;\
|
|
s0->header.impl_flags |= GRN_OBJ_EXPRVALUE;\
|
|
}\
|
|
} while (0)
|
|
|
|
#define POP2ALLOC1(arg1,arg2,value) do {\
|
|
if (EXPRVP(s0)) { vp--; }\
|
|
if (EXPRVP(s1)) { vp--; }\
|
|
arg2 = s0;\
|
|
arg1 = s1;\
|
|
sp--;\
|
|
if (sp < s_ + 1) { ERR(GRN_INVALID_ARGUMENT, "stack underflow"); goto exit; }\
|
|
s1 = sp[-2];\
|
|
sp[-1] = s0 = value = vp++;\
|
|
s0->header.impl_flags |= GRN_OBJ_EXPRVALUE;\
|
|
} while (0)
|
|
|
|
#define INTEGER_ARITHMETIC_OPERATION_PLUS(x, y) ((x) + (y))
|
|
#define FLOAT_ARITHMETIC_OPERATION_PLUS(x, y) ((double)(x) + (double)(y))
|
|
#define INTEGER_ARITHMETIC_OPERATION_MINUS(x, y) ((x) - (y))
|
|
#define FLOAT_ARITHMETIC_OPERATION_MINUS(x, y) ((double)(x) - (double)(y))
|
|
#define INTEGER_ARITHMETIC_OPERATION_STAR(x, y) ((x) * (y))
|
|
#define FLOAT_ARITHMETIC_OPERATION_STAR(x, y) ((double)(x) * (double)(y))
|
|
#define INTEGER_ARITHMETIC_OPERATION_SLASH(x, y) ((x) / (y))
|
|
#define FLOAT_ARITHMETIC_OPERATION_SLASH(x, y) ((double)(x) / (double)(y))
|
|
#define INTEGER_ARITHMETIC_OPERATION_MOD(x, y) ((x) % (y))
|
|
#define FLOAT_ARITHMETIC_OPERATION_MOD(x, y) (fmod((x), (y)))
|
|
#define INTEGER_ARITHMETIC_OPERATION_SHIFTL(x, y) ((x) << (y))
|
|
#define FLOAT_ARITHMETIC_OPERATION_SHIFTL(x, y) \
|
|
((long long int)(x) << (long long int)(y))
|
|
#define INTEGER_ARITHMETIC_OPERATION_SHIFTR(x, y) ((x) >> (y))
|
|
#define FLOAT_ARITHMETIC_OPERATION_SHIFTR(x, y) \
|
|
((long long int)(x) >> (long long int)(y))
|
|
#define INTEGER8_ARITHMETIC_OPERATION_SHIFTRR(x, y) \
|
|
((uint8_t)(x) >> (y))
|
|
#define INTEGER16_ARITHMETIC_OPERATION_SHIFTRR(x, y) \
|
|
((uint16_t)(x) >> (y))
|
|
#define INTEGER32_ARITHMETIC_OPERATION_SHIFTRR(x, y) \
|
|
((unsigned int)(x) >> (y))
|
|
#define INTEGER64_ARITHMETIC_OPERATION_SHIFTRR(x, y) \
|
|
((long long unsigned int)(x) >> (y))
|
|
#define FLOAT_ARITHMETIC_OPERATION_SHIFTRR(x, y) \
|
|
((long long unsigned int)(x) >> (long long unsigned int)(y))
|
|
|
|
#define INTEGER_ARITHMETIC_OPERATION_BITWISE_OR(x, y) ((x) | (y))
|
|
#define FLOAT_ARITHMETIC_OPERATION_BITWISE_OR(x, y) \
|
|
((long long int)(x) | (long long int)(y))
|
|
#define INTEGER_ARITHMETIC_OPERATION_BITWISE_XOR(x, y) ((x) ^ (y))
|
|
#define FLOAT_ARITHMETIC_OPERATION_BITWISE_XOR(x, y) \
|
|
((long long int)(x) ^ (long long int)(y))
|
|
#define INTEGER_ARITHMETIC_OPERATION_BITWISE_AND(x, y) ((x) & (y))
|
|
#define FLOAT_ARITHMETIC_OPERATION_BITWISE_AND(x, y) \
|
|
((long long int)(x) & (long long int)(y))
|
|
|
|
#define INTEGER_UNARY_ARITHMETIC_OPERATION_MINUS(x) (-(x))
|
|
#define FLOAT_UNARY_ARITHMETIC_OPERATION_MINUS(x) (-(x))
|
|
#define INTEGER_UNARY_ARITHMETIC_OPERATION_BITWISE_NOT(x) (~(x))
|
|
#define FLOAT_UNARY_ARITHMETIC_OPERATION_BITWISE_NOT(x) \
|
|
(~((long long int)(x)))
|
|
|
|
#define TEXT_ARITHMETIC_OPERATION(operator) do { \
|
|
long long int x_; \
|
|
long long int y_; \
|
|
\
|
|
res->header.domain = GRN_DB_INT64; \
|
|
\
|
|
GRN_INT64_SET(ctx, res, 0); \
|
|
grn_obj_cast(ctx, x, res, GRN_FALSE); \
|
|
x_ = GRN_INT64_VALUE(res); \
|
|
\
|
|
GRN_INT64_SET(ctx, res, 0); \
|
|
grn_obj_cast(ctx, y, res, GRN_FALSE); \
|
|
y_ = GRN_INT64_VALUE(res); \
|
|
\
|
|
GRN_INT64_SET(ctx, res, x_ operator y_); \
|
|
} while (0)
|
|
|
|
#define TEXT_UNARY_ARITHMETIC_OPERATION(unary_operator) do { \
|
|
long long int x_; \
|
|
\
|
|
res->header.domain = GRN_DB_INT64; \
|
|
\
|
|
GRN_INT64_SET(ctx, res, 0); \
|
|
grn_obj_cast(ctx, x, res, GRN_FALSE); \
|
|
x_ = GRN_INT64_VALUE(res); \
|
|
\
|
|
GRN_INT64_SET(ctx, res, unary_operator x_); \
|
|
} while (0)
|
|
|
|
#define ARITHMETIC_OPERATION_NO_CHECK(y) do {} while (0)
|
|
#define ARITHMETIC_OPERATION_ZERO_DIVISION_CHECK(y) do { \
|
|
if ((long long int)y == 0) { \
|
|
ERR(GRN_INVALID_ARGUMENT, "divisor should not be 0"); \
|
|
goto exit; \
|
|
} \
|
|
} while (0)
|
|
|
|
|
|
#define NUMERIC_ARITHMETIC_OPERATION_DISPATCH(set, get, x_, y, res, \
|
|
integer_operation, \
|
|
float_operation, \
|
|
right_expression_check, \
|
|
invalid_type_error) do { \
|
|
switch (y->header.domain) { \
|
|
case GRN_DB_INT8 : \
|
|
{ \
|
|
int8_t y_; \
|
|
y_ = GRN_INT8_VALUE(y); \
|
|
right_expression_check(y_); \
|
|
set(ctx, res, integer_operation(x_, y_)); \
|
|
} \
|
|
break; \
|
|
case GRN_DB_UINT8 : \
|
|
{ \
|
|
uint8_t y_; \
|
|
y_ = GRN_UINT8_VALUE(y); \
|
|
right_expression_check(y_); \
|
|
set(ctx, res, integer_operation(x_, y_)); \
|
|
} \
|
|
break; \
|
|
case GRN_DB_INT16 : \
|
|
{ \
|
|
int16_t y_; \
|
|
y_ = GRN_INT16_VALUE(y); \
|
|
right_expression_check(y_); \
|
|
set(ctx, res, integer_operation(x_, y_)); \
|
|
} \
|
|
break; \
|
|
case GRN_DB_UINT16 : \
|
|
{ \
|
|
uint16_t y_; \
|
|
y_ = GRN_UINT16_VALUE(y); \
|
|
right_expression_check(y_); \
|
|
set(ctx, res, integer_operation(x_, y_)); \
|
|
} \
|
|
break; \
|
|
case GRN_DB_INT32 : \
|
|
{ \
|
|
int y_; \
|
|
y_ = GRN_INT32_VALUE(y); \
|
|
right_expression_check(y_); \
|
|
set(ctx, res, integer_operation(x_, y_)); \
|
|
} \
|
|
break; \
|
|
case GRN_DB_UINT32 : \
|
|
{ \
|
|
unsigned int y_; \
|
|
y_ = GRN_UINT32_VALUE(y); \
|
|
right_expression_check(y_); \
|
|
set(ctx, res, integer_operation(x_, y_)); \
|
|
} \
|
|
break; \
|
|
case GRN_DB_TIME : \
|
|
{ \
|
|
long long int y_; \
|
|
y_ = GRN_TIME_VALUE(y); \
|
|
right_expression_check(y_); \
|
|
set(ctx, res, integer_operation(x_, y_)); \
|
|
} \
|
|
break; \
|
|
case GRN_DB_INT64 : \
|
|
{ \
|
|
long long int y_; \
|
|
y_ = GRN_INT64_VALUE(y); \
|
|
right_expression_check(y_); \
|
|
set(ctx, res, integer_operation(x_, y_)); \
|
|
} \
|
|
break; \
|
|
case GRN_DB_UINT64 : \
|
|
{ \
|
|
long long unsigned int y_; \
|
|
y_ = GRN_UINT64_VALUE(y); \
|
|
right_expression_check(y_); \
|
|
set(ctx, res, integer_operation(x_, y_)); \
|
|
} \
|
|
break; \
|
|
case GRN_DB_FLOAT : \
|
|
{ \
|
|
double y_; \
|
|
y_ = GRN_FLOAT_VALUE(y); \
|
|
right_expression_check(y_); \
|
|
res->header.domain = GRN_DB_FLOAT; \
|
|
GRN_FLOAT_SET(ctx, res, float_operation(x_, y_)); \
|
|
} \
|
|
break; \
|
|
case GRN_DB_SHORT_TEXT : \
|
|
case GRN_DB_TEXT : \
|
|
case GRN_DB_LONG_TEXT : \
|
|
set(ctx, res, 0); \
|
|
if (grn_obj_cast(ctx, y, res, GRN_FALSE)) { \
|
|
ERR(GRN_INVALID_ARGUMENT, \
|
|
"not a numerical format: <%.*s>", \
|
|
(int)GRN_TEXT_LEN(y), GRN_TEXT_VALUE(y)); \
|
|
goto exit; \
|
|
} \
|
|
set(ctx, res, integer_operation(x_, get(res))); \
|
|
break; \
|
|
default : \
|
|
invalid_type_error; \
|
|
break; \
|
|
} \
|
|
} while (0)
|
|
|
|
|
|
#define ARITHMETIC_OPERATION_DISPATCH(x, y, res, \
|
|
integer8_operation, \
|
|
integer16_operation, \
|
|
integer32_operation, \
|
|
integer64_operation, \
|
|
float_operation, \
|
|
left_expression_check, \
|
|
right_expression_check, \
|
|
text_operation, \
|
|
invalid_type_error) do { \
|
|
switch (x->header.domain) { \
|
|
case GRN_DB_INT8 : \
|
|
{ \
|
|
int8_t x_; \
|
|
x_ = GRN_INT8_VALUE(x); \
|
|
left_expression_check(x_); \
|
|
NUMERIC_ARITHMETIC_OPERATION_DISPATCH(GRN_INT8_SET, \
|
|
GRN_INT8_VALUE, \
|
|
x_, y, res, \
|
|
integer8_operation, \
|
|
float_operation, \
|
|
right_expression_check, \
|
|
invalid_type_error); \
|
|
} \
|
|
break; \
|
|
case GRN_DB_UINT8 : \
|
|
{ \
|
|
uint8_t x_; \
|
|
x_ = GRN_UINT8_VALUE(x); \
|
|
left_expression_check(x_); \
|
|
NUMERIC_ARITHMETIC_OPERATION_DISPATCH(GRN_UINT8_SET, \
|
|
GRN_UINT8_VALUE, \
|
|
x_, y, res, \
|
|
integer8_operation, \
|
|
float_operation, \
|
|
right_expression_check, \
|
|
invalid_type_error); \
|
|
} \
|
|
break; \
|
|
case GRN_DB_INT16 : \
|
|
{ \
|
|
int16_t x_; \
|
|
x_ = GRN_INT16_VALUE(x); \
|
|
left_expression_check(x_); \
|
|
NUMERIC_ARITHMETIC_OPERATION_DISPATCH(GRN_INT16_SET, \
|
|
GRN_INT16_VALUE, \
|
|
x_, y, res, \
|
|
integer16_operation, \
|
|
float_operation, \
|
|
right_expression_check, \
|
|
invalid_type_error); \
|
|
} \
|
|
break; \
|
|
case GRN_DB_UINT16 : \
|
|
{ \
|
|
uint16_t x_; \
|
|
x_ = GRN_UINT16_VALUE(x); \
|
|
left_expression_check(x_); \
|
|
NUMERIC_ARITHMETIC_OPERATION_DISPATCH(GRN_UINT16_SET, \
|
|
GRN_UINT16_VALUE, \
|
|
x_, y, res, \
|
|
integer16_operation, \
|
|
float_operation, \
|
|
right_expression_check, \
|
|
invalid_type_error); \
|
|
} \
|
|
break; \
|
|
case GRN_DB_INT32 : \
|
|
{ \
|
|
int x_; \
|
|
x_ = GRN_INT32_VALUE(x); \
|
|
left_expression_check(x_); \
|
|
NUMERIC_ARITHMETIC_OPERATION_DISPATCH(GRN_INT32_SET, \
|
|
GRN_INT32_VALUE, \
|
|
x_, y, res, \
|
|
integer32_operation, \
|
|
float_operation, \
|
|
right_expression_check, \
|
|
invalid_type_error); \
|
|
} \
|
|
break; \
|
|
case GRN_DB_UINT32 : \
|
|
{ \
|
|
unsigned int x_; \
|
|
x_ = GRN_UINT32_VALUE(x); \
|
|
left_expression_check(x_); \
|
|
NUMERIC_ARITHMETIC_OPERATION_DISPATCH(GRN_UINT32_SET, \
|
|
GRN_UINT32_VALUE, \
|
|
x_, y, res, \
|
|
integer32_operation, \
|
|
float_operation, \
|
|
right_expression_check, \
|
|
invalid_type_error); \
|
|
} \
|
|
break; \
|
|
case GRN_DB_INT64 : \
|
|
{ \
|
|
long long int x_; \
|
|
x_ = GRN_INT64_VALUE(x); \
|
|
left_expression_check(x_); \
|
|
NUMERIC_ARITHMETIC_OPERATION_DISPATCH(GRN_UINT64_SET, \
|
|
GRN_UINT64_VALUE, \
|
|
x_, y, res, \
|
|
integer64_operation, \
|
|
float_operation, \
|
|
right_expression_check, \
|
|
invalid_type_error); \
|
|
} \
|
|
break; \
|
|
case GRN_DB_TIME : \
|
|
{ \
|
|
long long int x_; \
|
|
x_ = GRN_TIME_VALUE(x); \
|
|
left_expression_check(x_); \
|
|
NUMERIC_ARITHMETIC_OPERATION_DISPATCH(GRN_TIME_SET, \
|
|
GRN_TIME_VALUE, \
|
|
x_, y, res, \
|
|
integer64_operation, \
|
|
float_operation, \
|
|
right_expression_check, \
|
|
invalid_type_error); \
|
|
} \
|
|
break; \
|
|
case GRN_DB_UINT64 : \
|
|
{ \
|
|
long long unsigned int x_; \
|
|
x_ = GRN_UINT64_VALUE(x); \
|
|
left_expression_check(x_); \
|
|
NUMERIC_ARITHMETIC_OPERATION_DISPATCH(GRN_UINT64_SET, \
|
|
GRN_UINT64_VALUE, \
|
|
x_, y, res, \
|
|
integer64_operation, \
|
|
float_operation, \
|
|
right_expression_check, \
|
|
invalid_type_error); \
|
|
} \
|
|
break; \
|
|
case GRN_DB_FLOAT : \
|
|
{ \
|
|
double x_; \
|
|
x_ = GRN_FLOAT_VALUE(x); \
|
|
left_expression_check(x_); \
|
|
NUMERIC_ARITHMETIC_OPERATION_DISPATCH(GRN_FLOAT_SET, \
|
|
GRN_FLOAT_VALUE, \
|
|
x_, y, res, \
|
|
float_operation, \
|
|
float_operation, \
|
|
right_expression_check, \
|
|
invalid_type_error); \
|
|
} \
|
|
break; \
|
|
case GRN_DB_SHORT_TEXT : \
|
|
case GRN_DB_TEXT : \
|
|
case GRN_DB_LONG_TEXT : \
|
|
text_operation; \
|
|
break; \
|
|
default: \
|
|
invalid_type_error; \
|
|
break; \
|
|
} \
|
|
code++; \
|
|
} while (0)
|
|
|
|
#define ARITHMETIC_BINARY_OPERATION_DISPATCH(operator, \
|
|
integer8_operation, \
|
|
integer16_operation, \
|
|
integer32_operation, \
|
|
integer64_operation, \
|
|
float_operation, \
|
|
left_expression_check, \
|
|
right_expression_check, \
|
|
text_operation, \
|
|
invalid_type_error) do { \
|
|
grn_obj *x, *y; \
|
|
\
|
|
POP2ALLOC1(x, y, res); \
|
|
if (x->header.type == GRN_VECTOR || y->header.type == GRN_VECTOR) { \
|
|
grn_obj inspected_x; \
|
|
grn_obj inspected_y; \
|
|
GRN_TEXT_INIT(&inspected_x, 0); \
|
|
GRN_TEXT_INIT(&inspected_y, 0); \
|
|
grn_inspect(ctx, &inspected_x, x); \
|
|
grn_inspect(ctx, &inspected_y, y); \
|
|
ERR(GRN_INVALID_ARGUMENT, \
|
|
"<%s> doesn't support vector: <%.*s> %s <%.*s>", \
|
|
operator, \
|
|
(int)GRN_TEXT_LEN(&inspected_x), GRN_TEXT_VALUE(&inspected_x), \
|
|
operator, \
|
|
(int)GRN_TEXT_LEN(&inspected_y), GRN_TEXT_VALUE(&inspected_y)); \
|
|
GRN_OBJ_FIN(ctx, &inspected_x); \
|
|
GRN_OBJ_FIN(ctx, &inspected_y); \
|
|
goto exit; \
|
|
} \
|
|
if (y != res) { \
|
|
res->header.domain = x->header.domain; \
|
|
} \
|
|
ARITHMETIC_OPERATION_DISPATCH(x, y, res, \
|
|
integer8_operation, \
|
|
integer16_operation, \
|
|
integer32_operation, \
|
|
integer64_operation, \
|
|
float_operation, \
|
|
left_expression_check, \
|
|
right_expression_check, \
|
|
text_operation, \
|
|
invalid_type_error); \
|
|
if (y == res) { \
|
|
res->header.domain = x->header.domain; \
|
|
} \
|
|
} while (0)
|
|
|
|
#define SIGNED_INTEGER_DIVISION_OPERATION_SLASH(x, y) \
|
|
((y == -1) ? -(x) : (x) / (y))
|
|
#define UNSIGNED_INTEGER_DIVISION_OPERATION_SLASH(x, y) ((x) / (y))
|
|
#define FLOAT_DIVISION_OPERATION_SLASH(x, y) ((double)(x) / (double)(y))
|
|
#define SIGNED_INTEGER_DIVISION_OPERATION_MOD(x, y) ((y == -1) ? 0 : (x) % (y))
|
|
#define UNSIGNED_INTEGER_DIVISION_OPERATION_MOD(x, y) ((x) % (y))
|
|
#define FLOAT_DIVISION_OPERATION_MOD(x, y) (fmod((x), (y)))
|
|
|
|
#define DIVISION_OPERATION_DISPATCH_RIGHT(set, get, x_, y, res, \
|
|
signed_integer_operation, \
|
|
unsigned_integer_operation, \
|
|
float_operation) do { \
|
|
switch (y->header.domain) { \
|
|
case GRN_DB_INT8 : \
|
|
{ \
|
|
int y_; \
|
|
y_ = GRN_INT8_VALUE(y); \
|
|
ARITHMETIC_OPERATION_ZERO_DIVISION_CHECK(y_); \
|
|
set(ctx, res, signed_integer_operation(x_, y_)); \
|
|
} \
|
|
break; \
|
|
case GRN_DB_UINT8 : \
|
|
{ \
|
|
int y_; \
|
|
y_ = GRN_UINT8_VALUE(y); \
|
|
ARITHMETIC_OPERATION_ZERO_DIVISION_CHECK(y_); \
|
|
set(ctx, res, signed_integer_operation(x_, y_)); \
|
|
} \
|
|
break; \
|
|
case GRN_DB_INT16 : \
|
|
{ \
|
|
int y_; \
|
|
y_ = GRN_INT16_VALUE(y); \
|
|
ARITHMETIC_OPERATION_ZERO_DIVISION_CHECK(y_); \
|
|
set(ctx, res, signed_integer_operation(x_, y_)); \
|
|
} \
|
|
break; \
|
|
case GRN_DB_UINT16 : \
|
|
{ \
|
|
int y_; \
|
|
y_ = GRN_UINT16_VALUE(y); \
|
|
ARITHMETIC_OPERATION_ZERO_DIVISION_CHECK(y_); \
|
|
set(ctx, res, signed_integer_operation(x_, y_)); \
|
|
} \
|
|
break; \
|
|
case GRN_DB_INT32 : \
|
|
{ \
|
|
int y_; \
|
|
y_ = GRN_INT32_VALUE(y); \
|
|
ARITHMETIC_OPERATION_ZERO_DIVISION_CHECK(y_); \
|
|
set(ctx, res, signed_integer_operation(x_, y_)); \
|
|
} \
|
|
break; \
|
|
case GRN_DB_UINT32 : \
|
|
{ \
|
|
unsigned int y_; \
|
|
y_ = GRN_UINT32_VALUE(y); \
|
|
ARITHMETIC_OPERATION_ZERO_DIVISION_CHECK(y_); \
|
|
set(ctx, res, unsigned_integer_operation(x_, y_)); \
|
|
} \
|
|
break; \
|
|
case GRN_DB_TIME : \
|
|
{ \
|
|
long long int y_; \
|
|
y_ = GRN_TIME_VALUE(y); \
|
|
ARITHMETIC_OPERATION_ZERO_DIVISION_CHECK(y_); \
|
|
set(ctx, res, signed_integer_operation(x_, y_)); \
|
|
} \
|
|
break; \
|
|
case GRN_DB_INT64 : \
|
|
{ \
|
|
long long int y_; \
|
|
y_ = GRN_INT64_VALUE(y); \
|
|
ARITHMETIC_OPERATION_ZERO_DIVISION_CHECK(y_); \
|
|
set(ctx, res, signed_integer_operation(x_, y_)); \
|
|
} \
|
|
break; \
|
|
case GRN_DB_UINT64 : \
|
|
{ \
|
|
long long unsigned int y_; \
|
|
y_ = GRN_UINT64_VALUE(y); \
|
|
ARITHMETIC_OPERATION_ZERO_DIVISION_CHECK(y_); \
|
|
set(ctx, res, unsigned_integer_operation(x_, y_)); \
|
|
} \
|
|
break; \
|
|
case GRN_DB_FLOAT : \
|
|
{ \
|
|
double y_; \
|
|
y_ = GRN_FLOAT_VALUE(y); \
|
|
ARITHMETIC_OPERATION_ZERO_DIVISION_CHECK(y_); \
|
|
res->header.domain = GRN_DB_FLOAT; \
|
|
GRN_FLOAT_SET(ctx, res, float_operation(x_, y_)); \
|
|
} \
|
|
break; \
|
|
case GRN_DB_SHORT_TEXT : \
|
|
case GRN_DB_TEXT : \
|
|
case GRN_DB_LONG_TEXT : \
|
|
set(ctx, res, 0); \
|
|
if (grn_obj_cast(ctx, y, res, GRN_FALSE)) { \
|
|
ERR(GRN_INVALID_ARGUMENT, \
|
|
"not a numerical format: <%.*s>", \
|
|
(int)GRN_TEXT_LEN(y), GRN_TEXT_VALUE(y)); \
|
|
goto exit; \
|
|
} \
|
|
/* The following "+ 0" is needed to suppress warnings that say */ \
|
|
/* comparison is always false due to limited range of data type */ \
|
|
set(ctx, res, signed_integer_operation(x_, (get(res) + 0))); \
|
|
break; \
|
|
default : \
|
|
break; \
|
|
} \
|
|
} while (0)
|
|
|
|
#define DIVISION_OPERATION_DISPATCH_LEFT(x, y, res, \
|
|
signed_integer_operation, \
|
|
unsigned_integer_operation, \
|
|
float_operation, \
|
|
invalid_type_error) do { \
|
|
switch (x->header.domain) { \
|
|
case GRN_DB_INT8 : \
|
|
{ \
|
|
int x_; \
|
|
x_ = GRN_INT8_VALUE(x); \
|
|
DIVISION_OPERATION_DISPATCH_RIGHT(GRN_INT8_SET, \
|
|
GRN_INT8_VALUE, \
|
|
x_, y, res, \
|
|
signed_integer_operation, \
|
|
unsigned_integer_operation, \
|
|
float_operation); \
|
|
} \
|
|
break; \
|
|
case GRN_DB_UINT8 : \
|
|
{ \
|
|
int x_; \
|
|
x_ = GRN_UINT8_VALUE(x); \
|
|
DIVISION_OPERATION_DISPATCH_RIGHT(GRN_UINT8_SET, \
|
|
(int)GRN_UINT8_VALUE, \
|
|
x_, y, res, \
|
|
signed_integer_operation, \
|
|
unsigned_integer_operation, \
|
|
float_operation); \
|
|
} \
|
|
break; \
|
|
case GRN_DB_INT16 : \
|
|
{ \
|
|
int x_; \
|
|
x_ = GRN_INT16_VALUE(x); \
|
|
DIVISION_OPERATION_DISPATCH_RIGHT(GRN_INT16_SET, \
|
|
GRN_INT16_VALUE, \
|
|
x_, y, res, \
|
|
signed_integer_operation, \
|
|
unsigned_integer_operation, \
|
|
float_operation); \
|
|
} \
|
|
break; \
|
|
case GRN_DB_UINT16 : \
|
|
{ \
|
|
int x_; \
|
|
x_ = GRN_UINT16_VALUE(x); \
|
|
DIVISION_OPERATION_DISPATCH_RIGHT(GRN_UINT16_SET, \
|
|
(int)GRN_UINT16_VALUE, \
|
|
x_, y, res, \
|
|
signed_integer_operation, \
|
|
unsigned_integer_operation, \
|
|
float_operation); \
|
|
} \
|
|
break; \
|
|
case GRN_DB_INT32 : \
|
|
{ \
|
|
int x_; \
|
|
x_ = GRN_INT32_VALUE(x); \
|
|
DIVISION_OPERATION_DISPATCH_RIGHT(GRN_INT32_SET, \
|
|
GRN_INT32_VALUE, \
|
|
x_, y, res, \
|
|
signed_integer_operation, \
|
|
unsigned_integer_operation, \
|
|
float_operation); \
|
|
} \
|
|
break; \
|
|
case GRN_DB_UINT32 : \
|
|
{ \
|
|
unsigned int x_; \
|
|
x_ = GRN_UINT32_VALUE(x); \
|
|
DIVISION_OPERATION_DISPATCH_RIGHT(GRN_UINT32_SET, \
|
|
GRN_UINT32_VALUE, \
|
|
x_, y, res, \
|
|
unsigned_integer_operation, \
|
|
unsigned_integer_operation, \
|
|
float_operation); \
|
|
} \
|
|
break; \
|
|
case GRN_DB_INT64 : \
|
|
{ \
|
|
long long int x_; \
|
|
x_ = GRN_INT64_VALUE(x); \
|
|
DIVISION_OPERATION_DISPATCH_RIGHT(GRN_INT64_SET, \
|
|
GRN_INT64_VALUE, \
|
|
x_, y, res, \
|
|
signed_integer_operation, \
|
|
unsigned_integer_operation, \
|
|
float_operation); \
|
|
} \
|
|
break; \
|
|
case GRN_DB_TIME : \
|
|
{ \
|
|
long long int x_; \
|
|
x_ = GRN_TIME_VALUE(x); \
|
|
DIVISION_OPERATION_DISPATCH_RIGHT(GRN_TIME_SET, \
|
|
GRN_TIME_VALUE, \
|
|
x_, y, res, \
|
|
signed_integer_operation, \
|
|
unsigned_integer_operation, \
|
|
float_operation); \
|
|
} \
|
|
break; \
|
|
case GRN_DB_UINT64 : \
|
|
{ \
|
|
long long unsigned int x_; \
|
|
x_ = GRN_UINT64_VALUE(x); \
|
|
DIVISION_OPERATION_DISPATCH_RIGHT(GRN_UINT64_SET, \
|
|
GRN_UINT64_VALUE, \
|
|
x_, y, res, \
|
|
unsigned_integer_operation, \
|
|
unsigned_integer_operation, \
|
|
float_operation); \
|
|
} \
|
|
break; \
|
|
case GRN_DB_FLOAT : \
|
|
{ \
|
|
double x_; \
|
|
x_ = GRN_FLOAT_VALUE(x); \
|
|
DIVISION_OPERATION_DISPATCH_RIGHT(GRN_FLOAT_SET, \
|
|
GRN_FLOAT_VALUE, \
|
|
x_, y, res, \
|
|
float_operation, \
|
|
float_operation, \
|
|
float_operation); \
|
|
} \
|
|
break; \
|
|
case GRN_DB_SHORT_TEXT : \
|
|
case GRN_DB_TEXT : \
|
|
case GRN_DB_LONG_TEXT : \
|
|
invalid_type_error; \
|
|
break; \
|
|
default: \
|
|
break; \
|
|
} \
|
|
code++; \
|
|
} while (0)
|
|
|
|
#define DIVISION_OPERATION_DISPATCH(signed_integer_operation, \
|
|
unsigned_integer_operation, \
|
|
float_operation, \
|
|
invalid_type_error) do { \
|
|
grn_obj *x, *y; \
|
|
\
|
|
POP2ALLOC1(x, y, res); \
|
|
if (y != res) { \
|
|
res->header.domain = x->header.domain; \
|
|
} \
|
|
DIVISION_OPERATION_DISPATCH_LEFT(x, y, res, \
|
|
signed_integer_operation, \
|
|
unsigned_integer_operation, \
|
|
float_operation, \
|
|
invalid_type_error); \
|
|
if (y == res) { \
|
|
res->header.domain = x->header.domain; \
|
|
} \
|
|
} while (0)
|
|
|
|
#define ARITHMETIC_UNARY_OPERATION_DISPATCH(integer_operation, \
|
|
float_operation, \
|
|
left_expression_check, \
|
|
right_expression_check, \
|
|
text_operation, \
|
|
invalid_type_error) do { \
|
|
grn_obj *x; \
|
|
POP1ALLOC1(x, res); \
|
|
res->header.domain = x->header.domain; \
|
|
switch (x->header.domain) { \
|
|
case GRN_DB_INT8 : \
|
|
{ \
|
|
int8_t x_; \
|
|
x_ = GRN_INT8_VALUE(x); \
|
|
left_expression_check(x_); \
|
|
GRN_INT8_SET(ctx, res, integer_operation(x_)); \
|
|
} \
|
|
break; \
|
|
case GRN_DB_UINT8 : \
|
|
{ \
|
|
int16_t x_; \
|
|
x_ = GRN_UINT8_VALUE(x); \
|
|
left_expression_check(x_); \
|
|
GRN_INT16_SET(ctx, res, integer_operation(x_)); \
|
|
res->header.domain = GRN_DB_INT16; \
|
|
} \
|
|
break; \
|
|
case GRN_DB_INT16 : \
|
|
{ \
|
|
int16_t x_; \
|
|
x_ = GRN_INT16_VALUE(x); \
|
|
left_expression_check(x_); \
|
|
GRN_INT16_SET(ctx, res, integer_operation(x_)); \
|
|
} \
|
|
break; \
|
|
case GRN_DB_UINT16 : \
|
|
{ \
|
|
int x_; \
|
|
x_ = GRN_UINT16_VALUE(x); \
|
|
left_expression_check(x_); \
|
|
GRN_INT32_SET(ctx, res, integer_operation(x_)); \
|
|
res->header.domain = GRN_DB_INT32; \
|
|
} \
|
|
break; \
|
|
case GRN_DB_INT32 : \
|
|
{ \
|
|
int x_; \
|
|
x_ = GRN_INT32_VALUE(x); \
|
|
left_expression_check(x_); \
|
|
GRN_INT32_SET(ctx, res, integer_operation(x_)); \
|
|
} \
|
|
break; \
|
|
case GRN_DB_UINT32 : \
|
|
{ \
|
|
long long int x_; \
|
|
x_ = GRN_UINT32_VALUE(x); \
|
|
left_expression_check(x_); \
|
|
GRN_INT64_SET(ctx, res, integer_operation(x_)); \
|
|
res->header.domain = GRN_DB_INT64; \
|
|
} \
|
|
break; \
|
|
case GRN_DB_INT64 : \
|
|
{ \
|
|
long long int x_; \
|
|
x_ = GRN_INT64_VALUE(x); \
|
|
left_expression_check(x_); \
|
|
GRN_INT64_SET(ctx, res, integer_operation(x_)); \
|
|
} \
|
|
break; \
|
|
case GRN_DB_TIME : \
|
|
{ \
|
|
long long int x_; \
|
|
x_ = GRN_TIME_VALUE(x); \
|
|
left_expression_check(x_); \
|
|
GRN_TIME_SET(ctx, res, integer_operation(x_)); \
|
|
} \
|
|
break; \
|
|
case GRN_DB_UINT64 : \
|
|
{ \
|
|
long long unsigned int x_; \
|
|
x_ = GRN_UINT64_VALUE(x); \
|
|
left_expression_check(x_); \
|
|
if (x_ > (long long unsigned int)INT64_MAX) { \
|
|
ERR(GRN_INVALID_ARGUMENT, \
|
|
"too large UInt64 value to inverse sign: " \
|
|
"<%" GRN_FMT_LLU ">", \
|
|
x_); \
|
|
goto exit; \
|
|
} else { \
|
|
long long int signed_x_; \
|
|
signed_x_ = x_; \
|
|
GRN_INT64_SET(ctx, res, integer_operation(signed_x_)); \
|
|
res->header.domain = GRN_DB_INT64; \
|
|
} \
|
|
} \
|
|
break; \
|
|
case GRN_DB_FLOAT : \
|
|
{ \
|
|
double x_; \
|
|
x_ = GRN_FLOAT_VALUE(x); \
|
|
left_expression_check(x_); \
|
|
GRN_FLOAT_SET(ctx, res, float_operation(x_)); \
|
|
} \
|
|
break; \
|
|
case GRN_DB_SHORT_TEXT : \
|
|
case GRN_DB_TEXT : \
|
|
case GRN_DB_LONG_TEXT : \
|
|
text_operation; \
|
|
break; \
|
|
default: \
|
|
invalid_type_error; \
|
|
break; \
|
|
} \
|
|
code++; \
|
|
} while (0)
|
|
|
|
#define EXEC_OPERATE(operate_sentence, assign_sentence) \
|
|
operate_sentence \
|
|
assign_sentence
|
|
|
|
#define EXEC_OPERATE_POST(operate_sentence, assign_sentence) \
|
|
assign_sentence \
|
|
operate_sentence
|
|
|
|
#define UNARY_OPERATE_AND_ASSIGN_DISPATCH(exec_operate, delta, \
|
|
set_flags) do { \
|
|
grn_obj *var, *col, value; \
|
|
grn_id rid; \
|
|
\
|
|
POP1ALLOC1(var, res); \
|
|
if (var->header.type != GRN_PTR) { \
|
|
ERR(GRN_INVALID_ARGUMENT, "invalid variable type: 0x%0x", \
|
|
var->header.type); \
|
|
goto exit; \
|
|
} \
|
|
if (GRN_BULK_VSIZE(var) != (sizeof(grn_obj *) + sizeof(grn_id))) { \
|
|
ERR(GRN_INVALID_ARGUMENT, \
|
|
"invalid variable size: " \
|
|
"expected: %" GRN_FMT_SIZE \
|
|
"actual: %" GRN_FMT_SIZE, \
|
|
(sizeof(grn_obj *) + sizeof(grn_id)), GRN_BULK_VSIZE(var)); \
|
|
goto exit; \
|
|
} \
|
|
col = GRN_PTR_VALUE(var); \
|
|
rid = *(grn_id *)(GRN_BULK_HEAD(var) + sizeof(grn_obj *)); \
|
|
res->header.type = GRN_VOID; \
|
|
res->header.domain = DB_OBJ(col)->range; \
|
|
switch (DB_OBJ(col)->range) { \
|
|
case GRN_DB_INT32 : \
|
|
GRN_INT32_INIT(&value, 0); \
|
|
GRN_INT32_SET(ctx, &value, delta); \
|
|
break; \
|
|
case GRN_DB_UINT32 : \
|
|
GRN_UINT32_INIT(&value, 0); \
|
|
GRN_UINT32_SET(ctx, &value, delta); \
|
|
break; \
|
|
case GRN_DB_INT64 : \
|
|
GRN_INT64_INIT(&value, 0); \
|
|
GRN_INT64_SET(ctx, &value, delta); \
|
|
break; \
|
|
case GRN_DB_UINT64 : \
|
|
GRN_UINT64_INIT(&value, 0); \
|
|
GRN_UINT64_SET(ctx, &value, delta); \
|
|
break; \
|
|
case GRN_DB_FLOAT : \
|
|
GRN_FLOAT_INIT(&value, 0); \
|
|
GRN_FLOAT_SET(ctx, &value, delta); \
|
|
break; \
|
|
case GRN_DB_TIME : \
|
|
GRN_TIME_INIT(&value, 0); \
|
|
GRN_TIME_SET(ctx, &value, GRN_TIME_PACK(delta, 0)); \
|
|
break; \
|
|
default: \
|
|
ERR(GRN_INVALID_ARGUMENT, \
|
|
"invalid increment target type: %d " \
|
|
"(FIXME: type name is needed)", DB_OBJ(col)->range); \
|
|
goto exit; \
|
|
break; \
|
|
} \
|
|
exec_operate(grn_obj_set_value(ctx, col, rid, &value, set_flags);, \
|
|
grn_obj_get_value(ctx, col, rid, res);); \
|
|
code++; \
|
|
} while (0)
|
|
|
|
#define ARITHMETIC_OPERATION_AND_ASSIGN_DISPATCH(integer8_operation, \
|
|
integer16_operation, \
|
|
integer32_operation, \
|
|
integer64_operation, \
|
|
float_operation, \
|
|
left_expression_check, \
|
|
right_expression_check,\
|
|
text_operation) do { \
|
|
grn_obj *value, *var, *res; \
|
|
if (code->value) { \
|
|
value = code->value; \
|
|
POP1ALLOC1(var, res); \
|
|
} else { \
|
|
POP2ALLOC1(var, value, res); \
|
|
} \
|
|
if (var->header.type == GRN_PTR && \
|
|
GRN_BULK_VSIZE(var) == (sizeof(grn_obj *) + sizeof(grn_id))) { \
|
|
grn_obj *col = GRN_PTR_VALUE(var); \
|
|
grn_id rid = *(grn_id *)(GRN_BULK_HEAD(var) + sizeof(grn_obj *)); \
|
|
grn_obj variable_value, casted_value; \
|
|
grn_id domain; \
|
|
\
|
|
value = GRN_OBJ_RESOLVE(ctx, value); \
|
|
\
|
|
domain = grn_obj_get_range(ctx, col); \
|
|
GRN_OBJ_INIT(&variable_value, GRN_BULK, 0, domain); \
|
|
grn_obj_get_value(ctx, col, rid, &variable_value); \
|
|
\
|
|
GRN_OBJ_INIT(&casted_value, GRN_BULK, 0, domain); \
|
|
if (grn_obj_cast(ctx, value, &casted_value, GRN_FALSE)) { \
|
|
ERR(GRN_INVALID_ARGUMENT, "invalid value: string"); \
|
|
GRN_OBJ_FIN(ctx, &variable_value); \
|
|
GRN_OBJ_FIN(ctx, &casted_value); \
|
|
POP1(res); \
|
|
goto exit; \
|
|
} \
|
|
grn_obj_reinit(ctx, res, domain, 0); \
|
|
ARITHMETIC_OPERATION_DISPATCH((&variable_value), (&casted_value), \
|
|
res, \
|
|
integer8_operation, \
|
|
integer16_operation, \
|
|
integer32_operation, \
|
|
integer64_operation, \
|
|
float_operation, \
|
|
left_expression_check, \
|
|
right_expression_check, \
|
|
text_operation,); \
|
|
grn_obj_set_value(ctx, col, rid, res, GRN_OBJ_SET); \
|
|
GRN_OBJ_FIN(ctx, (&variable_value)); \
|
|
GRN_OBJ_FIN(ctx, (&casted_value)); \
|
|
} else { \
|
|
ERR(GRN_INVALID_ARGUMENT, "left hand expression isn't column."); \
|
|
POP1(res); \
|
|
} \
|
|
} while (0)
|
|
|
|
static grn_bool
|
|
string_is_contained(grn_ctx *ctx,
|
|
const char *text, unsigned int text_len,
|
|
const char *sub_text, unsigned int sub_text_len)
|
|
{
|
|
/* TODO: Use more fast algorithm such as Boyer-Moore algorithm that
|
|
* is used in snip.c. */
|
|
const char *text_end = text + text_len;
|
|
unsigned int sub_text_current = 0;
|
|
|
|
for (; text < text_end; text++) {
|
|
if (text[0] == sub_text[sub_text_current]) {
|
|
sub_text_current++;
|
|
if (sub_text_current == sub_text_len) {
|
|
return GRN_TRUE;
|
|
}
|
|
} else {
|
|
sub_text_current = 0;
|
|
}
|
|
}
|
|
|
|
return GRN_FALSE;
|
|
}
|
|
|
|
static grn_bool
|
|
pseudo_query_scan_raw_text_raw_text(grn_ctx *ctx,
|
|
const char *x, unsigned int x_len,
|
|
const char *y, unsigned int y_len)
|
|
{
|
|
grn_obj *normalizer;
|
|
grn_obj *norm_x;
|
|
grn_obj *norm_y;
|
|
const char *norm_x_raw;
|
|
const char *norm_y_raw;
|
|
unsigned int norm_x_raw_length_in_bytes;
|
|
unsigned int norm_y_raw_length_in_bytes;
|
|
grn_bool matched = GRN_FALSE;
|
|
|
|
if (x_len == 0 || y_len == 0) {
|
|
return GRN_FALSE;
|
|
}
|
|
|
|
normalizer = grn_ctx_get(ctx, GRN_NORMALIZER_AUTO_NAME, -1);
|
|
norm_x = grn_string_open(ctx, x, x_len, normalizer, 0);
|
|
norm_y = grn_string_open(ctx, y, y_len, normalizer, 0);
|
|
grn_string_get_normalized(ctx, norm_x,
|
|
&norm_x_raw, &norm_x_raw_length_in_bytes,
|
|
NULL);
|
|
grn_string_get_normalized(ctx, norm_y,
|
|
&norm_y_raw, &norm_y_raw_length_in_bytes,
|
|
NULL);
|
|
matched = string_is_contained(ctx,
|
|
norm_x_raw, norm_x_raw_length_in_bytes,
|
|
norm_y_raw, norm_y_raw_length_in_bytes);
|
|
|
|
grn_obj_close(ctx, norm_x);
|
|
grn_obj_close(ctx, norm_y);
|
|
grn_obj_unlink(ctx, normalizer);
|
|
|
|
return matched;
|
|
}
|
|
|
|
static grn_bool
|
|
pseudo_query_scan_record_text(grn_ctx *ctx, grn_obj *record, grn_obj *table,
|
|
grn_obj *y)
|
|
{
|
|
grn_obj *normalizer;
|
|
char x_key[GRN_TABLE_MAX_KEY_SIZE];
|
|
int x_key_len;
|
|
grn_bool matched = GRN_FALSE;
|
|
|
|
if (table->header.domain != GRN_DB_SHORT_TEXT) {
|
|
return GRN_FALSE;
|
|
}
|
|
|
|
x_key_len = grn_table_get_key(ctx, table, GRN_RECORD_VALUE(record),
|
|
x_key, GRN_TABLE_MAX_KEY_SIZE);
|
|
grn_table_get_info(ctx, table, NULL, NULL, NULL, &normalizer, NULL);
|
|
if (normalizer) {
|
|
grn_obj *norm_y;
|
|
const char *norm_y_raw;
|
|
unsigned int norm_y_raw_length_in_bytes;
|
|
norm_y = grn_string_open(ctx, GRN_TEXT_VALUE(y), GRN_TEXT_LEN(y),
|
|
normalizer, 0);
|
|
grn_string_get_normalized(ctx, norm_y,
|
|
&norm_y_raw, &norm_y_raw_length_in_bytes,
|
|
NULL);
|
|
matched = string_is_contained(ctx,
|
|
x_key, x_key_len,
|
|
norm_y_raw, norm_y_raw_length_in_bytes);
|
|
grn_obj_close(ctx, norm_y);
|
|
} else {
|
|
matched = pseudo_query_scan_raw_text_raw_text(ctx,
|
|
x_key,
|
|
x_key_len,
|
|
GRN_TEXT_VALUE(y),
|
|
GRN_TEXT_LEN(y));
|
|
}
|
|
|
|
return matched;
|
|
}
|
|
|
|
static grn_bool
|
|
pseudo_query_scan_text_text(grn_ctx *ctx, grn_obj *x, grn_obj *y)
|
|
{
|
|
return pseudo_query_scan_raw_text_raw_text(ctx,
|
|
GRN_TEXT_VALUE(x),
|
|
GRN_TEXT_LEN(x),
|
|
GRN_TEXT_VALUE(y),
|
|
GRN_TEXT_LEN(y));
|
|
}
|
|
|
|
static grn_bool
|
|
pseudo_query_scan(grn_ctx *ctx, grn_obj *x, grn_obj *y)
|
|
{
|
|
switch (x->header.domain) {
|
|
case GRN_DB_SHORT_TEXT :
|
|
case GRN_DB_TEXT :
|
|
case GRN_DB_LONG_TEXT :
|
|
switch (y->header.domain) {
|
|
case GRN_DB_SHORT_TEXT :
|
|
case GRN_DB_TEXT :
|
|
case GRN_DB_LONG_TEXT :
|
|
return pseudo_query_scan_text_text(ctx, x, y);
|
|
default :
|
|
break;
|
|
}
|
|
return GRN_FALSE;
|
|
default:
|
|
{
|
|
grn_obj *domain;
|
|
domain = grn_ctx_at(ctx, x->header.domain);
|
|
if (GRN_OBJ_TABLEP(domain)) {
|
|
switch (y->header.domain) {
|
|
case GRN_DB_SHORT_TEXT :
|
|
case GRN_DB_TEXT :
|
|
case GRN_DB_LONG_TEXT :
|
|
return pseudo_query_scan_record_text(ctx, x, domain, y);
|
|
default :
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
return GRN_FALSE;
|
|
}
|
|
}
|
|
|
|
static grn_bool
|
|
pseudo_prefix_search_match(grn_ctx *ctx,
|
|
const char *x, unsigned int x_len,
|
|
const char *y, unsigned int y_len)
|
|
{
|
|
return (x_len >= y_len && strncmp(x, y, y_len) == 0);
|
|
}
|
|
|
|
static grn_bool
|
|
pseudo_prefix_search_raw_text_raw_text(grn_ctx *ctx,
|
|
const char *x, unsigned int x_len,
|
|
const char *y, unsigned int y_len)
|
|
{
|
|
grn_obj *normalizer;
|
|
grn_obj *norm_x;
|
|
grn_obj *norm_y;
|
|
const char *norm_x_raw;
|
|
const char *norm_y_raw;
|
|
unsigned int norm_x_raw_len;
|
|
unsigned int norm_y_raw_len;
|
|
grn_bool matched = GRN_FALSE;
|
|
|
|
normalizer = grn_ctx_get(ctx, GRN_NORMALIZER_AUTO_NAME, -1);
|
|
norm_x = grn_string_open(ctx, x, x_len, normalizer, 0);
|
|
norm_y = grn_string_open(ctx, y, y_len, normalizer, 0);
|
|
grn_string_get_normalized(ctx, norm_x, &norm_x_raw, &norm_x_raw_len, NULL);
|
|
grn_string_get_normalized(ctx, norm_y, &norm_y_raw, &norm_y_raw_len, NULL);
|
|
matched = pseudo_prefix_search_match(ctx,
|
|
norm_x_raw, norm_x_raw_len,
|
|
norm_y_raw, norm_y_raw_len);
|
|
|
|
grn_obj_close(ctx, norm_x);
|
|
grn_obj_close(ctx, norm_y);
|
|
grn_obj_unlink(ctx, normalizer);
|
|
|
|
return matched;
|
|
}
|
|
|
|
static grn_bool
|
|
pseudo_prefix_search_record_text(grn_ctx *ctx, grn_obj *record, grn_obj *table,
|
|
grn_obj *y)
|
|
{
|
|
grn_obj *normalizer;
|
|
char x_key[GRN_TABLE_MAX_KEY_SIZE];
|
|
int x_key_len;
|
|
grn_bool matched = GRN_FALSE;
|
|
|
|
if (table->header.domain != GRN_DB_SHORT_TEXT) {
|
|
return GRN_FALSE;
|
|
}
|
|
|
|
x_key_len = grn_table_get_key(ctx, table, GRN_RECORD_VALUE(record),
|
|
x_key, GRN_TABLE_MAX_KEY_SIZE);
|
|
grn_table_get_info(ctx, table, NULL, NULL, NULL, &normalizer, NULL);
|
|
if (normalizer) {
|
|
grn_obj *norm_y;
|
|
const char *norm_y_raw;
|
|
unsigned int norm_y_raw_len;
|
|
norm_y = grn_string_open(ctx, GRN_TEXT_VALUE(y), GRN_TEXT_LEN(y),
|
|
normalizer, 0);
|
|
grn_string_get_normalized(ctx, norm_y, &norm_y_raw, &norm_y_raw_len, NULL);
|
|
matched = pseudo_prefix_search_match(ctx,
|
|
x_key, x_key_len,
|
|
norm_y_raw, norm_y_raw_len);
|
|
grn_obj_close(ctx, norm_y);
|
|
} else {
|
|
matched = pseudo_prefix_search_raw_text_raw_text(ctx,
|
|
x_key,
|
|
x_key_len,
|
|
GRN_TEXT_VALUE(y),
|
|
GRN_TEXT_LEN(y));
|
|
}
|
|
|
|
return matched;
|
|
}
|
|
|
|
static grn_bool
|
|
pseudo_prefix_search_text_text(grn_ctx *ctx, grn_obj *x, grn_obj *y)
|
|
{
|
|
return pseudo_prefix_search_raw_text_raw_text(ctx,
|
|
GRN_TEXT_VALUE(x),
|
|
GRN_TEXT_LEN(x),
|
|
GRN_TEXT_VALUE(y),
|
|
GRN_TEXT_LEN(y));
|
|
}
|
|
|
|
static grn_bool
|
|
pseudo_prefix_search(grn_ctx *ctx, grn_obj *x, grn_obj *y)
|
|
{
|
|
switch (x->header.domain) {
|
|
case GRN_DB_SHORT_TEXT :
|
|
case GRN_DB_TEXT :
|
|
case GRN_DB_LONG_TEXT :
|
|
switch (y->header.domain) {
|
|
case GRN_DB_SHORT_TEXT :
|
|
case GRN_DB_TEXT :
|
|
case GRN_DB_LONG_TEXT :
|
|
return pseudo_prefix_search_text_text(ctx, x, y);
|
|
default :
|
|
break;
|
|
}
|
|
return GRN_FALSE;
|
|
default:
|
|
{
|
|
grn_obj *domain;
|
|
domain = grn_ctx_at(ctx, x->header.domain);
|
|
if (GRN_OBJ_TABLEP(domain)) {
|
|
switch (y->header.domain) {
|
|
case GRN_DB_SHORT_TEXT :
|
|
case GRN_DB_TEXT :
|
|
case GRN_DB_LONG_TEXT :
|
|
return pseudo_prefix_search_record_text(ctx, x, domain, y);
|
|
default :
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
return GRN_FALSE;
|
|
}
|
|
}
|
|
|
|
inline static void
|
|
grn_expr_exec_get_member(grn_ctx *ctx,
|
|
grn_obj *expr,
|
|
grn_obj *column_and_record_id,
|
|
grn_obj *index,
|
|
grn_obj *result)
|
|
{
|
|
grn_obj *column;
|
|
grn_id record_id;
|
|
grn_obj values;
|
|
int i;
|
|
|
|
column = GRN_PTR_VALUE(column_and_record_id);
|
|
record_id = *((grn_id *)(&(GRN_PTR_VALUE_AT(column_and_record_id, 1))));
|
|
GRN_TEXT_INIT(&values, 0);
|
|
grn_obj_get_value(ctx, column, record_id, &values);
|
|
|
|
i = GRN_UINT32_VALUE(index);
|
|
if (values.header.type == GRN_UVECTOR) {
|
|
int n_elements;
|
|
grn_obj_reinit(ctx, result, DB_OBJ(column)->range, 0);
|
|
n_elements = GRN_BULK_VSIZE(&values) / sizeof(grn_id);
|
|
if (n_elements > i) {
|
|
grn_id value;
|
|
value = GRN_RECORD_VALUE_AT(&values, i);
|
|
GRN_RECORD_SET(ctx, result, value);
|
|
}
|
|
} else {
|
|
if (values.u.v.n_sections > i) {
|
|
const char *content;
|
|
unsigned int content_length;
|
|
grn_id domain;
|
|
|
|
content_length = grn_vector_get_element(ctx, &values, i,
|
|
&content, NULL, &domain);
|
|
grn_obj_reinit(ctx, result, domain, 0);
|
|
grn_bulk_write(ctx, result, content, content_length);
|
|
}
|
|
}
|
|
|
|
GRN_OBJ_FIN(ctx, &values);
|
|
}
|
|
|
|
grn_obj *
|
|
grn_expr_exec(grn_ctx *ctx, grn_obj *expr, int nargs)
|
|
{
|
|
grn_obj *val = NULL;
|
|
uint32_t stack_curr = ctx->impl->stack_curr;
|
|
GRN_API_ENTER;
|
|
if (expr->header.type == GRN_PROC) {
|
|
grn_proc *proc = (grn_proc *)expr;
|
|
if (proc->type == GRN_PROC_COMMAND) {
|
|
grn_command_input *input;
|
|
input = grn_command_input_open(ctx, expr);
|
|
grn_command_run(ctx, expr, input);
|
|
grn_command_input_close(ctx, input);
|
|
GRN_API_RETURN(NULL);
|
|
} else {
|
|
grn_proc_call(ctx, expr, nargs, expr);
|
|
}
|
|
} else {
|
|
grn_expr *e = (grn_expr *)expr;
|
|
register grn_obj **s_ = ctx->impl->stack, *s0 = NULL, *s1 = NULL, **sp, *vp = e->values;
|
|
grn_obj *res = NULL, *v0 = grn_expr_get_var_by_offset(ctx, expr, 0);
|
|
grn_expr_code *code = e->codes, *ce = &e->codes[e->codes_curr];
|
|
sp = s_ + stack_curr;
|
|
while (code < ce) {
|
|
switch (code->op) {
|
|
case GRN_OP_NOP :
|
|
code++;
|
|
break;
|
|
case GRN_OP_PUSH :
|
|
PUSH1(code->value);
|
|
code++;
|
|
break;
|
|
case GRN_OP_POP :
|
|
{
|
|
grn_obj *obj;
|
|
POP1(obj);
|
|
code++;
|
|
}
|
|
break;
|
|
case GRN_OP_GET_REF :
|
|
{
|
|
grn_obj *col, *rec;
|
|
if (code->nargs == 1) {
|
|
rec = v0;
|
|
if (code->value) {
|
|
col = code->value;
|
|
ALLOC1(res);
|
|
} else {
|
|
POP1ALLOC1(col, res);
|
|
}
|
|
} else {
|
|
if (code->value) {
|
|
col = code->value;
|
|
POP1ALLOC1(rec, res);
|
|
} else {
|
|
POP2ALLOC1(rec, col, res);
|
|
}
|
|
}
|
|
if (col->header.type == GRN_BULK) {
|
|
grn_obj *table = grn_ctx_at(ctx, GRN_OBJ_GET_DOMAIN(rec));
|
|
col = grn_obj_column(ctx, table, GRN_BULK_HEAD(col), GRN_BULK_VSIZE(col));
|
|
if (col) { grn_expr_take_obj(ctx, (grn_obj *)e, col); }
|
|
}
|
|
if (col) {
|
|
res->header.type = GRN_PTR;
|
|
res->header.domain = GRN_ID_NIL;
|
|
GRN_PTR_SET(ctx, res, col);
|
|
GRN_UINT32_PUT(ctx, res, GRN_RECORD_VALUE(rec));
|
|
} else {
|
|
ERR(GRN_INVALID_ARGUMENT, "col resolve failed");
|
|
goto exit;
|
|
}
|
|
code++;
|
|
}
|
|
break;
|
|
case GRN_OP_CALL :
|
|
{
|
|
grn_obj *proc;
|
|
if (code->value) {
|
|
if (sp < s_ + code->nargs) {
|
|
ERR(GRN_INVALID_ARGUMENT, "stack error");
|
|
goto exit;
|
|
}
|
|
proc = code->value;
|
|
WITH_SPSAVE({
|
|
grn_proc_call(ctx, proc, code->nargs, expr);
|
|
});
|
|
} else {
|
|
int offset = code->nargs + 1;
|
|
if (sp < s_ + offset) {
|
|
ERR(GRN_INVALID_ARGUMENT, "stack error");
|
|
goto exit;
|
|
}
|
|
proc = sp[-offset];
|
|
WITH_SPSAVE({
|
|
grn_proc_call(ctx, proc, code->nargs, expr);
|
|
});
|
|
if (ctx->rc) {
|
|
goto exit;
|
|
}
|
|
POP1(res);
|
|
{
|
|
grn_obj *proc_;
|
|
POP1(proc_);
|
|
if (proc != proc_) {
|
|
GRN_LOG(ctx, GRN_LOG_WARNING, "stack may be corrupt");
|
|
}
|
|
}
|
|
PUSH1(res);
|
|
}
|
|
}
|
|
code++;
|
|
break;
|
|
case GRN_OP_INTERN :
|
|
{
|
|
grn_obj *obj;
|
|
POP1(obj);
|
|
obj = GRN_OBJ_RESOLVE(ctx, obj);
|
|
res = grn_expr_get_var(ctx, expr, GRN_TEXT_VALUE(obj), GRN_TEXT_LEN(obj));
|
|
if (!res) { res = grn_ctx_get(ctx, GRN_TEXT_VALUE(obj), GRN_TEXT_LEN(obj)); }
|
|
if (!res) {
|
|
ERR(GRN_INVALID_ARGUMENT, "intern failed");
|
|
goto exit;
|
|
}
|
|
PUSH1(res);
|
|
}
|
|
code++;
|
|
break;
|
|
case GRN_OP_TABLE_CREATE :
|
|
{
|
|
grn_obj *value_type, *key_type, *flags, *name;
|
|
POP1(value_type);
|
|
value_type = GRN_OBJ_RESOLVE(ctx, value_type);
|
|
POP1(key_type);
|
|
key_type = GRN_OBJ_RESOLVE(ctx, key_type);
|
|
POP1(flags);
|
|
flags = GRN_OBJ_RESOLVE(ctx, flags);
|
|
POP1(name);
|
|
name = GRN_OBJ_RESOLVE(ctx, name);
|
|
res = grn_table_create(ctx, GRN_TEXT_VALUE(name), GRN_TEXT_LEN(name),
|
|
NULL, GRN_UINT32_VALUE(flags),
|
|
key_type, value_type);
|
|
PUSH1(res);
|
|
}
|
|
code++;
|
|
break;
|
|
case GRN_OP_EXPR_GET_VAR :
|
|
{
|
|
grn_obj *name, *expr;
|
|
POP1(name);
|
|
name = GRN_OBJ_RESOLVE(ctx, name);
|
|
POP1(expr);
|
|
expr = GRN_OBJ_RESOLVE(ctx, expr);
|
|
switch (name->header.domain) {
|
|
case GRN_DB_INT32 :
|
|
res = grn_expr_get_var_by_offset(ctx, expr, (unsigned int) GRN_INT32_VALUE(name));
|
|
break;
|
|
case GRN_DB_UINT32 :
|
|
res = grn_expr_get_var_by_offset(ctx, expr, (unsigned int) GRN_UINT32_VALUE(name));
|
|
break;
|
|
case GRN_DB_INT64 :
|
|
res = grn_expr_get_var_by_offset(ctx, expr, (unsigned int) GRN_INT64_VALUE(name));
|
|
break;
|
|
case GRN_DB_UINT64 :
|
|
res = grn_expr_get_var_by_offset(ctx, expr, (unsigned int) GRN_UINT64_VALUE(name));
|
|
break;
|
|
case GRN_DB_SHORT_TEXT :
|
|
case GRN_DB_TEXT :
|
|
case GRN_DB_LONG_TEXT :
|
|
res = grn_expr_get_var(ctx, expr, GRN_TEXT_VALUE(name), GRN_TEXT_LEN(name));
|
|
break;
|
|
default :
|
|
ERR(GRN_INVALID_ARGUMENT, "invalid type");
|
|
goto exit;
|
|
}
|
|
PUSH1(res);
|
|
}
|
|
code++;
|
|
break;
|
|
case GRN_OP_ASSIGN :
|
|
{
|
|
grn_obj *value, *var;
|
|
if (code->value) {
|
|
value = code->value;
|
|
} else {
|
|
POP1(value);
|
|
}
|
|
value = GRN_OBJ_RESOLVE(ctx, value);
|
|
POP1(var);
|
|
// var = GRN_OBJ_RESOLVE(ctx, var);
|
|
if (var->header.type == GRN_PTR &&
|
|
GRN_BULK_VSIZE(var) == (sizeof(grn_obj *) + sizeof(grn_id))) {
|
|
grn_obj *col = GRN_PTR_VALUE(var);
|
|
grn_id rid = *(grn_id *)(GRN_BULK_HEAD(var) + sizeof(grn_obj *));
|
|
grn_obj_set_value(ctx, col, rid, value, GRN_OBJ_SET);
|
|
} else {
|
|
VAR_SET_VALUE(ctx, var, value);
|
|
}
|
|
PUSH1(value);
|
|
}
|
|
code++;
|
|
break;
|
|
case GRN_OP_STAR_ASSIGN :
|
|
ARITHMETIC_OPERATION_AND_ASSIGN_DISPATCH(
|
|
INTEGER_ARITHMETIC_OPERATION_STAR,
|
|
INTEGER_ARITHMETIC_OPERATION_STAR,
|
|
INTEGER_ARITHMETIC_OPERATION_STAR,
|
|
INTEGER_ARITHMETIC_OPERATION_STAR,
|
|
FLOAT_ARITHMETIC_OPERATION_STAR,
|
|
ARITHMETIC_OPERATION_NO_CHECK,
|
|
ARITHMETIC_OPERATION_NO_CHECK,
|
|
{
|
|
ERR(GRN_INVALID_ARGUMENT, "variable *= \"string\" isn't supported");
|
|
goto exit;
|
|
});
|
|
break;
|
|
case GRN_OP_SLASH_ASSIGN :
|
|
ARITHMETIC_OPERATION_AND_ASSIGN_DISPATCH(
|
|
INTEGER_ARITHMETIC_OPERATION_SLASH,
|
|
INTEGER_ARITHMETIC_OPERATION_SLASH,
|
|
INTEGER_ARITHMETIC_OPERATION_SLASH,
|
|
INTEGER_ARITHMETIC_OPERATION_SLASH,
|
|
FLOAT_ARITHMETIC_OPERATION_SLASH,
|
|
ARITHMETIC_OPERATION_NO_CHECK,
|
|
ARITHMETIC_OPERATION_NO_CHECK,
|
|
{
|
|
ERR(GRN_INVALID_ARGUMENT, "variable /= \"string\" isn't supported");
|
|
goto exit;
|
|
});
|
|
break;
|
|
case GRN_OP_MOD_ASSIGN :
|
|
ARITHMETIC_OPERATION_AND_ASSIGN_DISPATCH(
|
|
INTEGER_ARITHMETIC_OPERATION_MOD,
|
|
INTEGER_ARITHMETIC_OPERATION_MOD,
|
|
INTEGER_ARITHMETIC_OPERATION_MOD,
|
|
INTEGER_ARITHMETIC_OPERATION_MOD,
|
|
FLOAT_ARITHMETIC_OPERATION_MOD,
|
|
ARITHMETIC_OPERATION_NO_CHECK,
|
|
ARITHMETIC_OPERATION_NO_CHECK,
|
|
{
|
|
ERR(GRN_INVALID_ARGUMENT, "variable %%= \"string\" isn't supported");
|
|
goto exit;
|
|
});
|
|
break;
|
|
case GRN_OP_PLUS_ASSIGN :
|
|
ARITHMETIC_OPERATION_AND_ASSIGN_DISPATCH(
|
|
INTEGER_ARITHMETIC_OPERATION_PLUS,
|
|
INTEGER_ARITHMETIC_OPERATION_PLUS,
|
|
INTEGER_ARITHMETIC_OPERATION_PLUS,
|
|
INTEGER_ARITHMETIC_OPERATION_PLUS,
|
|
FLOAT_ARITHMETIC_OPERATION_PLUS,
|
|
ARITHMETIC_OPERATION_NO_CHECK,
|
|
ARITHMETIC_OPERATION_NO_CHECK,
|
|
{
|
|
ERR(GRN_INVALID_ARGUMENT, "variable += \"string\" isn't supported");
|
|
goto exit;
|
|
});
|
|
break;
|
|
case GRN_OP_MINUS_ASSIGN :
|
|
ARITHMETIC_OPERATION_AND_ASSIGN_DISPATCH(
|
|
INTEGER_ARITHMETIC_OPERATION_MINUS,
|
|
INTEGER_ARITHMETIC_OPERATION_MINUS,
|
|
INTEGER_ARITHMETIC_OPERATION_MINUS,
|
|
INTEGER_ARITHMETIC_OPERATION_MINUS,
|
|
FLOAT_ARITHMETIC_OPERATION_MINUS,
|
|
ARITHMETIC_OPERATION_NO_CHECK,
|
|
ARITHMETIC_OPERATION_NO_CHECK,
|
|
{
|
|
ERR(GRN_INVALID_ARGUMENT, "variable -= \"string\" isn't supported");
|
|
goto exit;
|
|
});
|
|
break;
|
|
case GRN_OP_SHIFTL_ASSIGN :
|
|
ARITHMETIC_OPERATION_AND_ASSIGN_DISPATCH(
|
|
INTEGER_ARITHMETIC_OPERATION_SHIFTL,
|
|
INTEGER_ARITHMETIC_OPERATION_SHIFTL,
|
|
INTEGER_ARITHMETIC_OPERATION_SHIFTL,
|
|
INTEGER_ARITHMETIC_OPERATION_SHIFTL,
|
|
FLOAT_ARITHMETIC_OPERATION_SHIFTL,
|
|
ARITHMETIC_OPERATION_NO_CHECK,
|
|
ARITHMETIC_OPERATION_NO_CHECK,
|
|
{
|
|
ERR(GRN_INVALID_ARGUMENT, "variable <<= \"string\" isn't supported");
|
|
goto exit;
|
|
});
|
|
break;
|
|
case GRN_OP_SHIFTR_ASSIGN :
|
|
ARITHMETIC_OPERATION_AND_ASSIGN_DISPATCH(
|
|
INTEGER_ARITHMETIC_OPERATION_SHIFTR,
|
|
INTEGER_ARITHMETIC_OPERATION_SHIFTR,
|
|
INTEGER_ARITHMETIC_OPERATION_SHIFTR,
|
|
INTEGER_ARITHMETIC_OPERATION_SHIFTR,
|
|
FLOAT_ARITHMETIC_OPERATION_SHIFTR,
|
|
ARITHMETIC_OPERATION_NO_CHECK,
|
|
ARITHMETIC_OPERATION_NO_CHECK,
|
|
{
|
|
ERR(GRN_INVALID_ARGUMENT, "variable >>= \"string\" isn't supported");
|
|
goto exit;
|
|
});
|
|
break;
|
|
case GRN_OP_SHIFTRR_ASSIGN :
|
|
ARITHMETIC_OPERATION_AND_ASSIGN_DISPATCH(
|
|
INTEGER8_ARITHMETIC_OPERATION_SHIFTRR,
|
|
INTEGER16_ARITHMETIC_OPERATION_SHIFTRR,
|
|
INTEGER32_ARITHMETIC_OPERATION_SHIFTRR,
|
|
INTEGER64_ARITHMETIC_OPERATION_SHIFTRR,
|
|
FLOAT_ARITHMETIC_OPERATION_SHIFTRR,
|
|
ARITHMETIC_OPERATION_NO_CHECK,
|
|
ARITHMETIC_OPERATION_NO_CHECK,
|
|
{
|
|
ERR(GRN_INVALID_ARGUMENT,
|
|
"variable >>>= \"string\" isn't supported");
|
|
goto exit;
|
|
});
|
|
break;
|
|
case GRN_OP_AND_ASSIGN :
|
|
ARITHMETIC_OPERATION_AND_ASSIGN_DISPATCH(
|
|
INTEGER_ARITHMETIC_OPERATION_BITWISE_AND,
|
|
INTEGER_ARITHMETIC_OPERATION_BITWISE_AND,
|
|
INTEGER_ARITHMETIC_OPERATION_BITWISE_AND,
|
|
INTEGER_ARITHMETIC_OPERATION_BITWISE_AND,
|
|
FLOAT_ARITHMETIC_OPERATION_BITWISE_AND,
|
|
ARITHMETIC_OPERATION_NO_CHECK,
|
|
ARITHMETIC_OPERATION_NO_CHECK,
|
|
{
|
|
ERR(GRN_INVALID_ARGUMENT, "variable &= \"string\" isn't supported");
|
|
goto exit;
|
|
});
|
|
break;
|
|
case GRN_OP_OR_ASSIGN :
|
|
ARITHMETIC_OPERATION_AND_ASSIGN_DISPATCH(
|
|
INTEGER_ARITHMETIC_OPERATION_BITWISE_OR,
|
|
INTEGER_ARITHMETIC_OPERATION_BITWISE_OR,
|
|
INTEGER_ARITHMETIC_OPERATION_BITWISE_OR,
|
|
INTEGER_ARITHMETIC_OPERATION_BITWISE_OR,
|
|
FLOAT_ARITHMETIC_OPERATION_BITWISE_OR,
|
|
ARITHMETIC_OPERATION_NO_CHECK,
|
|
ARITHMETIC_OPERATION_NO_CHECK,
|
|
{
|
|
ERR(GRN_INVALID_ARGUMENT, "variable |= \"string\" isn't supported");
|
|
goto exit;
|
|
});
|
|
break;
|
|
case GRN_OP_XOR_ASSIGN :
|
|
ARITHMETIC_OPERATION_AND_ASSIGN_DISPATCH(
|
|
INTEGER_ARITHMETIC_OPERATION_BITWISE_XOR,
|
|
INTEGER_ARITHMETIC_OPERATION_BITWISE_XOR,
|
|
INTEGER_ARITHMETIC_OPERATION_BITWISE_XOR,
|
|
INTEGER_ARITHMETIC_OPERATION_BITWISE_XOR,
|
|
FLOAT_ARITHMETIC_OPERATION_BITWISE_XOR,
|
|
ARITHMETIC_OPERATION_NO_CHECK,
|
|
ARITHMETIC_OPERATION_NO_CHECK,
|
|
{
|
|
ERR(GRN_INVALID_ARGUMENT, "variable ^= \"string\" isn't supported");
|
|
goto exit;
|
|
});
|
|
break;
|
|
case GRN_OP_JUMP :
|
|
code += code->nargs + 1;
|
|
break;
|
|
case GRN_OP_CJUMP :
|
|
{
|
|
grn_obj *v;
|
|
unsigned int v_boolean;
|
|
POP1(v);
|
|
GRN_TRUEP(ctx, v, v_boolean);
|
|
if (!v_boolean) { code += code->nargs; }
|
|
}
|
|
code++;
|
|
break;
|
|
case GRN_OP_GET_VALUE :
|
|
{
|
|
grn_obj *col, *rec;
|
|
do {
|
|
if (code->nargs == 1) {
|
|
rec = v0;
|
|
if (code->value) {
|
|
col = code->value;
|
|
ALLOC1(res);
|
|
} else {
|
|
POP1ALLOC1(col, res);
|
|
}
|
|
} else {
|
|
if (code->value) {
|
|
col = code->value;
|
|
POP1ALLOC1(rec, res);
|
|
} else {
|
|
POP2ALLOC1(rec, col, res);
|
|
}
|
|
}
|
|
if (col->header.type == GRN_BULK) {
|
|
grn_obj *table = grn_ctx_at(ctx, GRN_OBJ_GET_DOMAIN(rec));
|
|
col = grn_obj_column(ctx, table, GRN_BULK_HEAD(col), GRN_BULK_VSIZE(col));
|
|
if (col) { grn_expr_take_obj(ctx, (grn_obj *)expr, col); }
|
|
}
|
|
if (!col) {
|
|
ERR(GRN_INVALID_ARGUMENT, "col resolve failed");
|
|
goto exit;
|
|
}
|
|
grn_obj_reinit_for(ctx, res, col);
|
|
grn_obj_get_value(ctx, col, GRN_RECORD_VALUE(rec), res);
|
|
code++;
|
|
} while (code < ce && code->op == GRN_OP_GET_VALUE);
|
|
}
|
|
break;
|
|
case GRN_OP_OBJ_SEARCH :
|
|
{
|
|
grn_obj *op, *query, *index;
|
|
// todo : grn_search_optarg optarg;
|
|
POP1(op);
|
|
op = GRN_OBJ_RESOLVE(ctx, op);
|
|
POP1(res);
|
|
res = GRN_OBJ_RESOLVE(ctx, res);
|
|
POP1(query);
|
|
query = GRN_OBJ_RESOLVE(ctx, query);
|
|
POP1(index);
|
|
index = GRN_OBJ_RESOLVE(ctx, index);
|
|
grn_obj_search(ctx, index, query, res,
|
|
(grn_operator)GRN_UINT32_VALUE(op), NULL);
|
|
}
|
|
code++;
|
|
break;
|
|
case GRN_OP_TABLE_SELECT :
|
|
{
|
|
grn_obj *op, *res, *expr, *table;
|
|
POP1(op);
|
|
op = GRN_OBJ_RESOLVE(ctx, op);
|
|
POP1(res);
|
|
res = GRN_OBJ_RESOLVE(ctx, res);
|
|
POP1(expr);
|
|
expr = GRN_OBJ_RESOLVE(ctx, expr);
|
|
POP1(table);
|
|
table = GRN_OBJ_RESOLVE(ctx, table);
|
|
WITH_SPSAVE({
|
|
grn_table_select(ctx, table, expr, res, (grn_operator)GRN_UINT32_VALUE(op));
|
|
});
|
|
PUSH1(res);
|
|
}
|
|
code++;
|
|
break;
|
|
case GRN_OP_TABLE_SORT :
|
|
{
|
|
grn_obj *keys_, *res, *limit, *table;
|
|
POP1(keys_);
|
|
keys_ = GRN_OBJ_RESOLVE(ctx, keys_);
|
|
POP1(res);
|
|
res = GRN_OBJ_RESOLVE(ctx, res);
|
|
POP1(limit);
|
|
limit = GRN_OBJ_RESOLVE(ctx, limit);
|
|
POP1(table);
|
|
table = GRN_OBJ_RESOLVE(ctx, table);
|
|
{
|
|
grn_table_sort_key *keys;
|
|
const char *p = GRN_BULK_HEAD(keys_), *tokbuf[256];
|
|
int n = grn_str_tok(p, GRN_BULK_VSIZE(keys_), ' ', tokbuf, 256, NULL);
|
|
if ((keys = GRN_MALLOCN(grn_table_sort_key, n))) {
|
|
int i, n_keys = 0;
|
|
for (i = 0; i < n; i++) {
|
|
uint32_t len = (uint32_t) (tokbuf[i] - p);
|
|
grn_obj *col = grn_obj_column(ctx, table, p, len);
|
|
if (col) {
|
|
keys[n_keys].key = col;
|
|
keys[n_keys].flags = GRN_TABLE_SORT_ASC;
|
|
keys[n_keys].offset = 0;
|
|
n_keys++;
|
|
} else {
|
|
if (p[0] == ':' && p[1] == 'd' && len == 2 && n_keys) {
|
|
keys[n_keys - 1].flags |= GRN_TABLE_SORT_DESC;
|
|
}
|
|
}
|
|
p = tokbuf[i] + 1;
|
|
}
|
|
WITH_SPSAVE({
|
|
grn_table_sort(ctx, table, 0, GRN_INT32_VALUE(limit), res, keys, n_keys);
|
|
});
|
|
for (i = 0; i < n_keys; i++) {
|
|
grn_obj_unlink(ctx, keys[i].key);
|
|
}
|
|
GRN_FREE(keys);
|
|
}
|
|
}
|
|
}
|
|
code++;
|
|
break;
|
|
case GRN_OP_TABLE_GROUP :
|
|
{
|
|
grn_obj *res, *keys_, *table;
|
|
POP1(res);
|
|
res = GRN_OBJ_RESOLVE(ctx, res);
|
|
POP1(keys_);
|
|
keys_ = GRN_OBJ_RESOLVE(ctx, keys_);
|
|
POP1(table);
|
|
table = GRN_OBJ_RESOLVE(ctx, table);
|
|
{
|
|
grn_table_sort_key *keys;
|
|
grn_table_group_result results;
|
|
const char *p = GRN_BULK_HEAD(keys_), *tokbuf[256];
|
|
int n = grn_str_tok(p, GRN_BULK_VSIZE(keys_), ' ', tokbuf, 256, NULL);
|
|
if ((keys = GRN_MALLOCN(grn_table_sort_key, n))) {
|
|
int i, n_keys = 0;
|
|
for (i = 0; i < n; i++) {
|
|
uint32_t len = (uint32_t) (tokbuf[i] - p);
|
|
grn_obj *col = grn_obj_column(ctx, table, p, len);
|
|
if (col) {
|
|
keys[n_keys].key = col;
|
|
keys[n_keys].flags = GRN_TABLE_SORT_ASC;
|
|
keys[n_keys].offset = 0;
|
|
n_keys++;
|
|
} else if (n_keys) {
|
|
if (p[0] == ':' && p[1] == 'd' && len == 2) {
|
|
keys[n_keys - 1].flags |= GRN_TABLE_SORT_DESC;
|
|
} else {
|
|
keys[n_keys - 1].offset = grn_atoi(p, p + len, NULL);
|
|
}
|
|
}
|
|
p = tokbuf[i] + 1;
|
|
}
|
|
/* todo : support multi-results */
|
|
results.table = res;
|
|
results.key_begin = 0;
|
|
results.key_end = 0;
|
|
results.limit = 0;
|
|
results.flags = 0;
|
|
results.op = GRN_OP_OR;
|
|
WITH_SPSAVE({
|
|
grn_table_group(ctx, table, keys, n_keys, &results, 1);
|
|
});
|
|
for (i = 0; i < n_keys; i++) {
|
|
grn_obj_unlink(ctx, keys[i].key);
|
|
}
|
|
GRN_FREE(keys);
|
|
}
|
|
}
|
|
}
|
|
code++;
|
|
break;
|
|
case GRN_OP_JSON_PUT :
|
|
{
|
|
grn_obj_format format;
|
|
grn_obj *str, *table, *res;
|
|
POP1(res);
|
|
res = GRN_OBJ_RESOLVE(ctx, res);
|
|
POP1(str);
|
|
str = GRN_OBJ_RESOLVE(ctx, str);
|
|
POP1(table);
|
|
table = GRN_OBJ_RESOLVE(ctx, table);
|
|
GRN_OBJ_FORMAT_INIT(&format, grn_table_size(ctx, table), 0, -1, 0);
|
|
format.flags = 0;
|
|
grn_obj_columns(ctx, table,
|
|
GRN_TEXT_VALUE(str), GRN_TEXT_LEN(str), &format.columns);
|
|
grn_text_otoj(ctx, res, table, &format);
|
|
GRN_OBJ_FORMAT_FIN(ctx, &format);
|
|
}
|
|
code++;
|
|
break;
|
|
case GRN_OP_AND :
|
|
{
|
|
grn_obj *x, *y;
|
|
unsigned int x_boolean, y_boolean;
|
|
grn_obj *result = NULL;
|
|
POP2ALLOC1(x, y, res);
|
|
GRN_TRUEP(ctx, x, x_boolean);
|
|
if (x_boolean) {
|
|
GRN_TRUEP(ctx, y, y_boolean);
|
|
if (y_boolean) {
|
|
result = y;
|
|
}
|
|
}
|
|
if (result) {
|
|
if (res != result) {
|
|
grn_obj_reinit(ctx, res, result->header.domain, 0);
|
|
grn_obj_cast(ctx, result, res, GRN_FALSE);
|
|
}
|
|
} else {
|
|
grn_obj_reinit(ctx, res, GRN_DB_BOOL, 0);
|
|
GRN_BOOL_SET(ctx, res, GRN_FALSE);
|
|
}
|
|
}
|
|
code++;
|
|
break;
|
|
case GRN_OP_OR :
|
|
{
|
|
grn_obj *x, *y;
|
|
unsigned int x_boolean, y_boolean;
|
|
grn_obj *result;
|
|
POP2ALLOC1(x, y, res);
|
|
GRN_TRUEP(ctx, x, x_boolean);
|
|
if (x_boolean) {
|
|
result = x;
|
|
} else {
|
|
GRN_TRUEP(ctx, y, y_boolean);
|
|
if (y_boolean) {
|
|
result = y;
|
|
} else {
|
|
result = NULL;
|
|
}
|
|
}
|
|
if (result) {
|
|
if (res != result) {
|
|
grn_obj_reinit(ctx, res, result->header.domain, 0);
|
|
grn_obj_cast(ctx, result, res, GRN_FALSE);
|
|
}
|
|
} else {
|
|
grn_obj_reinit(ctx, res, GRN_DB_BOOL, 0);
|
|
GRN_BOOL_SET(ctx, res, GRN_FALSE);
|
|
}
|
|
}
|
|
code++;
|
|
break;
|
|
case GRN_OP_AND_NOT :
|
|
{
|
|
grn_obj *x, *y;
|
|
POP2ALLOC1(x, y, res);
|
|
if (GRN_INT32_VALUE(x) == 0 || GRN_INT32_VALUE(y) == 1) {
|
|
GRN_INT32_SET(ctx, res, 0);
|
|
} else {
|
|
GRN_INT32_SET(ctx, res, 1);
|
|
}
|
|
res->header.type = GRN_BULK;
|
|
res->header.domain = GRN_DB_INT32;
|
|
}
|
|
code++;
|
|
break;
|
|
case GRN_OP_ADJUST :
|
|
{
|
|
/* todo */
|
|
}
|
|
code++;
|
|
break;
|
|
case GRN_OP_MATCH :
|
|
{
|
|
grn_obj *x, *y;
|
|
grn_bool matched;
|
|
POP1(y);
|
|
POP1(x);
|
|
WITH_SPSAVE({
|
|
matched = pseudo_query_scan(ctx, x, y);
|
|
});
|
|
ALLOC1(res);
|
|
grn_obj_reinit(ctx, res, GRN_DB_INT32, 0);
|
|
GRN_INT32_SET(ctx, res, matched ? 1 : 0);
|
|
}
|
|
code++;
|
|
break;
|
|
case GRN_OP_EQUAL :
|
|
{
|
|
grn_bool is_equal;
|
|
grn_obj *x, *y;
|
|
POP2ALLOC1(x, y, res);
|
|
is_equal = grn_operator_exec_equal(ctx, x, y);
|
|
grn_obj_reinit(ctx, res, GRN_DB_INT32, 0);
|
|
GRN_INT32_SET(ctx, res, is_equal ? 1 : 0);
|
|
}
|
|
code++;
|
|
break;
|
|
case GRN_OP_NOT_EQUAL :
|
|
{
|
|
grn_bool is_not_equal;
|
|
grn_obj *x, *y;
|
|
POP2ALLOC1(x, y, res);
|
|
is_not_equal = grn_operator_exec_not_equal(ctx, x, y);
|
|
grn_obj_reinit(ctx, res, GRN_DB_INT32, 0);
|
|
GRN_INT32_SET(ctx, res, is_not_equal ? 1 : 0);
|
|
}
|
|
code++;
|
|
break;
|
|
case GRN_OP_PREFIX :
|
|
{
|
|
grn_obj *x, *y;
|
|
grn_bool matched;
|
|
POP1(y);
|
|
POP1(x);
|
|
WITH_SPSAVE({
|
|
matched = pseudo_prefix_search(ctx, x, y);
|
|
});
|
|
ALLOC1(res);
|
|
grn_obj_reinit(ctx, res, GRN_DB_INT32, 0);
|
|
GRN_INT32_SET(ctx, res, matched ? 1 : 0);
|
|
}
|
|
code++;
|
|
break;
|
|
case GRN_OP_SUFFIX :
|
|
{
|
|
grn_obj *x, *y;
|
|
POP2ALLOC1(x, y, res);
|
|
GRN_INT32_SET(ctx, res,
|
|
(GRN_TEXT_LEN(x) >= GRN_TEXT_LEN(y) &&
|
|
!memcmp(GRN_TEXT_VALUE(x) + GRN_TEXT_LEN(x) - GRN_TEXT_LEN(y),
|
|
GRN_TEXT_VALUE(y), GRN_TEXT_LEN(y))));
|
|
res->header.type = GRN_BULK;
|
|
res->header.domain = GRN_DB_INT32;
|
|
}
|
|
code++;
|
|
break;
|
|
case GRN_OP_LESS :
|
|
{
|
|
grn_bool r;
|
|
grn_obj *x, *y;
|
|
POP2ALLOC1(x, y, res);
|
|
r = grn_operator_exec_less(ctx, x, y);
|
|
GRN_INT32_SET(ctx, res, r ? 1 : 0);
|
|
res->header.type = GRN_BULK;
|
|
res->header.domain = GRN_DB_INT32;
|
|
}
|
|
code++;
|
|
break;
|
|
case GRN_OP_GREATER :
|
|
{
|
|
grn_bool r;
|
|
grn_obj *x, *y;
|
|
POP2ALLOC1(x, y, res);
|
|
r = grn_operator_exec_greater(ctx, x, y);
|
|
GRN_INT32_SET(ctx, res, r ? 1 : 0);
|
|
res->header.type = GRN_BULK;
|
|
res->header.domain = GRN_DB_INT32;
|
|
}
|
|
code++;
|
|
break;
|
|
case GRN_OP_LESS_EQUAL :
|
|
{
|
|
grn_bool r;
|
|
grn_obj *x, *y;
|
|
POP2ALLOC1(x, y, res);
|
|
r = grn_operator_exec_less_equal(ctx, x, y);
|
|
GRN_INT32_SET(ctx, res, r ? 1 : 0);
|
|
res->header.type = GRN_BULK;
|
|
res->header.domain = GRN_DB_INT32;
|
|
}
|
|
code++;
|
|
break;
|
|
case GRN_OP_GREATER_EQUAL :
|
|
{
|
|
grn_bool r;
|
|
grn_obj *x, *y;
|
|
POP2ALLOC1(x, y, res);
|
|
r = grn_operator_exec_greater_equal(ctx, x, y);
|
|
GRN_INT32_SET(ctx, res, r ? 1 : 0);
|
|
res->header.type = GRN_BULK;
|
|
res->header.domain = GRN_DB_INT32;
|
|
}
|
|
code++;
|
|
break;
|
|
case GRN_OP_GEO_DISTANCE1 :
|
|
{
|
|
grn_obj *e;
|
|
double lng1, lat1, lng2, lat2, x, y, d;
|
|
POP1(e);
|
|
lng1 = GEO_INT2RAD(GRN_INT32_VALUE(e));
|
|
POP1(e);
|
|
lat1 = GEO_INT2RAD(GRN_INT32_VALUE(e));
|
|
POP1(e);
|
|
lng2 = GEO_INT2RAD(GRN_INT32_VALUE(e));
|
|
POP1ALLOC1(e, res);
|
|
lat2 = GEO_INT2RAD(GRN_INT32_VALUE(e));
|
|
x = (lng2 - lng1) * cos((lat1 + lat2) * 0.5);
|
|
y = (lat2 - lat1);
|
|
d = sqrt((x * x) + (y * y)) * GEO_RADIOUS;
|
|
res->header.type = GRN_BULK;
|
|
res->header.domain = GRN_DB_FLOAT;
|
|
GRN_FLOAT_SET(ctx, res, d);
|
|
}
|
|
code++;
|
|
break;
|
|
case GRN_OP_GEO_DISTANCE2 :
|
|
{
|
|
grn_obj *e;
|
|
double lng1, lat1, lng2, lat2, x, y, d;
|
|
POP1(e);
|
|
lng1 = GEO_INT2RAD(GRN_INT32_VALUE(e));
|
|
POP1(e);
|
|
lat1 = GEO_INT2RAD(GRN_INT32_VALUE(e));
|
|
POP1(e);
|
|
lng2 = GEO_INT2RAD(GRN_INT32_VALUE(e));
|
|
POP1ALLOC1(e, res);
|
|
lat2 = GEO_INT2RAD(GRN_INT32_VALUE(e));
|
|
x = sin(fabs(lng2 - lng1) * 0.5);
|
|
y = sin(fabs(lat2 - lat1) * 0.5);
|
|
d = asin(sqrt((y * y) + cos(lat1) * cos(lat2) * x * x)) * 2 * GEO_RADIOUS;
|
|
res->header.type = GRN_BULK;
|
|
res->header.domain = GRN_DB_FLOAT;
|
|
GRN_FLOAT_SET(ctx, res, d);
|
|
}
|
|
code++;
|
|
break;
|
|
case GRN_OP_GEO_DISTANCE3 :
|
|
{
|
|
grn_obj *e;
|
|
double lng1, lat1, lng2, lat2, p, q, m, n, x, y, d;
|
|
POP1(e);
|
|
lng1 = GEO_INT2RAD(GRN_INT32_VALUE(e));
|
|
POP1(e);
|
|
lat1 = GEO_INT2RAD(GRN_INT32_VALUE(e));
|
|
POP1(e);
|
|
lng2 = GEO_INT2RAD(GRN_INT32_VALUE(e));
|
|
POP1ALLOC1(e, res);
|
|
lat2 = GEO_INT2RAD(GRN_INT32_VALUE(e));
|
|
p = (lat1 + lat2) * 0.5;
|
|
q = (1 - GEO_BES_C3 * sin(p) * sin(p));
|
|
m = GEO_BES_C1 / sqrt(q * q * q);
|
|
n = GEO_BES_C2 / sqrt(q);
|
|
x = n * cos(p) * fabs(lng1 - lng2);
|
|
y = m * fabs(lat1 - lat2);
|
|
d = sqrt((x * x) + (y * y));
|
|
res->header.type = GRN_BULK;
|
|
res->header.domain = GRN_DB_FLOAT;
|
|
GRN_FLOAT_SET(ctx, res, d);
|
|
}
|
|
code++;
|
|
break;
|
|
case GRN_OP_GEO_DISTANCE4 :
|
|
{
|
|
grn_obj *e;
|
|
double lng1, lat1, lng2, lat2, p, q, m, n, x, y, d;
|
|
POP1(e);
|
|
lng1 = GEO_INT2RAD(GRN_INT32_VALUE(e));
|
|
POP1(e);
|
|
lat1 = GEO_INT2RAD(GRN_INT32_VALUE(e));
|
|
POP1(e);
|
|
lng2 = GEO_INT2RAD(GRN_INT32_VALUE(e));
|
|
POP1ALLOC1(e, res);
|
|
lat2 = GEO_INT2RAD(GRN_INT32_VALUE(e));
|
|
p = (lat1 + lat2) * 0.5;
|
|
q = (1 - GEO_GRS_C3 * sin(p) * sin(p));
|
|
m = GEO_GRS_C1 / sqrt(q * q * q);
|
|
n = GEO_GRS_C2 / sqrt(q);
|
|
x = n * cos(p) * fabs(lng1 - lng2);
|
|
y = m * fabs(lat1 - lat2);
|
|
d = sqrt((x * x) + (y * y));
|
|
res->header.type = GRN_BULK;
|
|
res->header.domain = GRN_DB_FLOAT;
|
|
GRN_FLOAT_SET(ctx, res, d);
|
|
}
|
|
code++;
|
|
break;
|
|
case GRN_OP_GEO_WITHINP5 :
|
|
{
|
|
int r;
|
|
grn_obj *e;
|
|
double lng0, lat0, lng1, lat1, x, y, d;
|
|
POP1(e);
|
|
lng0 = GEO_INT2RAD(GRN_INT32_VALUE(e));
|
|
POP1(e);
|
|
lat0 = GEO_INT2RAD(GRN_INT32_VALUE(e));
|
|
POP1(e);
|
|
lng1 = GEO_INT2RAD(GRN_INT32_VALUE(e));
|
|
POP1(e);
|
|
lat1 = GEO_INT2RAD(GRN_INT32_VALUE(e));
|
|
POP1ALLOC1(e, res);
|
|
x = (lng1 - lng0) * cos((lat0 + lat1) * 0.5);
|
|
y = (lat1 - lat0);
|
|
d = sqrt((x * x) + (y * y)) * GEO_RADIOUS;
|
|
switch (e->header.domain) {
|
|
case GRN_DB_INT32 :
|
|
r = d <= GRN_INT32_VALUE(e);
|
|
break;
|
|
case GRN_DB_FLOAT :
|
|
r = d <= GRN_FLOAT_VALUE(e);
|
|
break;
|
|
default :
|
|
r = 0;
|
|
break;
|
|
}
|
|
GRN_INT32_SET(ctx, res, r);
|
|
res->header.type = GRN_BULK;
|
|
res->header.domain = GRN_DB_INT32;
|
|
}
|
|
code++;
|
|
break;
|
|
case GRN_OP_GEO_WITHINP6 :
|
|
{
|
|
int r;
|
|
grn_obj *e;
|
|
double lng0, lat0, lng1, lat1, lng2, lat2, x, y, d;
|
|
POP1(e);
|
|
lng0 = GEO_INT2RAD(GRN_INT32_VALUE(e));
|
|
POP1(e);
|
|
lat0 = GEO_INT2RAD(GRN_INT32_VALUE(e));
|
|
POP1(e);
|
|
lng1 = GEO_INT2RAD(GRN_INT32_VALUE(e));
|
|
POP1(e);
|
|
lat1 = GEO_INT2RAD(GRN_INT32_VALUE(e));
|
|
POP1(e);
|
|
lng2 = GEO_INT2RAD(GRN_INT32_VALUE(e));
|
|
POP1ALLOC1(e, res);
|
|
lat2 = GEO_INT2RAD(GRN_INT32_VALUE(e));
|
|
x = (lng1 - lng0) * cos((lat0 + lat1) * 0.5);
|
|
y = (lat1 - lat0);
|
|
d = (x * x) + (y * y);
|
|
x = (lng2 - lng1) * cos((lat1 + lat2) * 0.5);
|
|
y = (lat2 - lat1);
|
|
r = d <= (x * x) + (y * y);
|
|
GRN_INT32_SET(ctx, res, r);
|
|
res->header.type = GRN_BULK;
|
|
res->header.domain = GRN_DB_INT32;
|
|
}
|
|
code++;
|
|
break;
|
|
case GRN_OP_GEO_WITHINP8 :
|
|
{
|
|
int r;
|
|
grn_obj *e;
|
|
int64_t ln0, la0, ln1, la1, ln2, la2, ln3, la3;
|
|
POP1(e);
|
|
ln0 = GRN_INT32_VALUE(e);
|
|
POP1(e);
|
|
la0 = GRN_INT32_VALUE(e);
|
|
POP1(e);
|
|
ln1 = GRN_INT32_VALUE(e);
|
|
POP1(e);
|
|
la1 = GRN_INT32_VALUE(e);
|
|
POP1(e);
|
|
ln2 = GRN_INT32_VALUE(e);
|
|
POP1(e);
|
|
la2 = GRN_INT32_VALUE(e);
|
|
POP1(e);
|
|
ln3 = GRN_INT32_VALUE(e);
|
|
POP1ALLOC1(e, res);
|
|
la3 = GRN_INT32_VALUE(e);
|
|
r = ((ln2 <= ln0) && (ln0 <= ln3) && (la2 <= la0) && (la0 <= la3));
|
|
GRN_INT32_SET(ctx, res, r);
|
|
res->header.type = GRN_BULK;
|
|
res->header.domain = GRN_DB_INT32;
|
|
}
|
|
code++;
|
|
break;
|
|
case GRN_OP_PLUS :
|
|
ARITHMETIC_BINARY_OPERATION_DISPATCH(
|
|
"+",
|
|
INTEGER_ARITHMETIC_OPERATION_PLUS,
|
|
INTEGER_ARITHMETIC_OPERATION_PLUS,
|
|
INTEGER_ARITHMETIC_OPERATION_PLUS,
|
|
INTEGER_ARITHMETIC_OPERATION_PLUS,
|
|
FLOAT_ARITHMETIC_OPERATION_PLUS,
|
|
ARITHMETIC_OPERATION_NO_CHECK,
|
|
ARITHMETIC_OPERATION_NO_CHECK,
|
|
{
|
|
if (x == res) {
|
|
grn_obj_cast(ctx, y, res, GRN_FALSE);
|
|
} else if (y == res) {
|
|
grn_obj buffer;
|
|
GRN_TEXT_INIT(&buffer, 0);
|
|
grn_obj_cast(ctx, x, &buffer, GRN_FALSE);
|
|
grn_obj_cast(ctx, y, &buffer, GRN_FALSE);
|
|
GRN_BULK_REWIND(res);
|
|
grn_obj_cast(ctx, &buffer, res, GRN_FALSE);
|
|
GRN_OBJ_FIN(ctx, &buffer);
|
|
} else {
|
|
GRN_BULK_REWIND(res);
|
|
grn_obj_cast(ctx, x, res, GRN_FALSE);
|
|
grn_obj_cast(ctx, y, res, GRN_FALSE);
|
|
}
|
|
}
|
|
,);
|
|
break;
|
|
case GRN_OP_MINUS :
|
|
if (code->nargs == 1) {
|
|
ARITHMETIC_UNARY_OPERATION_DISPATCH(
|
|
INTEGER_UNARY_ARITHMETIC_OPERATION_MINUS,
|
|
FLOAT_UNARY_ARITHMETIC_OPERATION_MINUS,
|
|
ARITHMETIC_OPERATION_NO_CHECK,
|
|
ARITHMETIC_OPERATION_NO_CHECK,
|
|
{
|
|
long long int x_;
|
|
|
|
res->header.type = GRN_BULK;
|
|
res->header.domain = GRN_DB_INT64;
|
|
|
|
GRN_INT64_SET(ctx, res, 0);
|
|
grn_obj_cast(ctx, x, res, GRN_FALSE);
|
|
x_ = GRN_INT64_VALUE(res);
|
|
|
|
GRN_INT64_SET(ctx, res, -x_);
|
|
}
|
|
,);
|
|
} else {
|
|
ARITHMETIC_BINARY_OPERATION_DISPATCH(
|
|
"-",
|
|
INTEGER_ARITHMETIC_OPERATION_MINUS,
|
|
INTEGER_ARITHMETIC_OPERATION_MINUS,
|
|
INTEGER_ARITHMETIC_OPERATION_MINUS,
|
|
INTEGER_ARITHMETIC_OPERATION_MINUS,
|
|
FLOAT_ARITHMETIC_OPERATION_MINUS,
|
|
ARITHMETIC_OPERATION_NO_CHECK,
|
|
ARITHMETIC_OPERATION_NO_CHECK,
|
|
{
|
|
ERR(GRN_INVALID_ARGUMENT,
|
|
"\"string\" - \"string\" "
|
|
"isn't supported");
|
|
goto exit;
|
|
}
|
|
,);
|
|
}
|
|
break;
|
|
case GRN_OP_STAR :
|
|
ARITHMETIC_BINARY_OPERATION_DISPATCH(
|
|
"*",
|
|
INTEGER_ARITHMETIC_OPERATION_STAR,
|
|
INTEGER_ARITHMETIC_OPERATION_STAR,
|
|
INTEGER_ARITHMETIC_OPERATION_STAR,
|
|
INTEGER_ARITHMETIC_OPERATION_STAR,
|
|
FLOAT_ARITHMETIC_OPERATION_STAR,
|
|
ARITHMETIC_OPERATION_NO_CHECK,
|
|
ARITHMETIC_OPERATION_NO_CHECK,
|
|
{
|
|
ERR(GRN_INVALID_ARGUMENT,
|
|
"\"string\" * \"string\" "
|
|
"isn't supported");
|
|
goto exit;
|
|
}
|
|
,);
|
|
break;
|
|
case GRN_OP_SLASH :
|
|
DIVISION_OPERATION_DISPATCH(
|
|
SIGNED_INTEGER_DIVISION_OPERATION_SLASH,
|
|
UNSIGNED_INTEGER_DIVISION_OPERATION_SLASH,
|
|
FLOAT_DIVISION_OPERATION_SLASH,
|
|
{
|
|
ERR(GRN_INVALID_ARGUMENT,
|
|
"\"string\" / \"string\" "
|
|
"isn't supported");
|
|
goto exit;
|
|
});
|
|
break;
|
|
case GRN_OP_MOD :
|
|
DIVISION_OPERATION_DISPATCH(
|
|
SIGNED_INTEGER_DIVISION_OPERATION_MOD,
|
|
UNSIGNED_INTEGER_DIVISION_OPERATION_MOD,
|
|
FLOAT_DIVISION_OPERATION_MOD,
|
|
{
|
|
ERR(GRN_INVALID_ARGUMENT,
|
|
"\"string\" %% \"string\" "
|
|
"isn't supported");
|
|
goto exit;
|
|
});
|
|
break;
|
|
case GRN_OP_BITWISE_NOT :
|
|
ARITHMETIC_UNARY_OPERATION_DISPATCH(
|
|
INTEGER_UNARY_ARITHMETIC_OPERATION_BITWISE_NOT,
|
|
FLOAT_UNARY_ARITHMETIC_OPERATION_BITWISE_NOT,
|
|
ARITHMETIC_OPERATION_NO_CHECK,
|
|
ARITHMETIC_OPERATION_NO_CHECK,
|
|
TEXT_UNARY_ARITHMETIC_OPERATION(~),);
|
|
break;
|
|
case GRN_OP_BITWISE_OR :
|
|
ARITHMETIC_BINARY_OPERATION_DISPATCH(
|
|
"|",
|
|
INTEGER_ARITHMETIC_OPERATION_BITWISE_OR,
|
|
INTEGER_ARITHMETIC_OPERATION_BITWISE_OR,
|
|
INTEGER_ARITHMETIC_OPERATION_BITWISE_OR,
|
|
INTEGER_ARITHMETIC_OPERATION_BITWISE_OR,
|
|
FLOAT_ARITHMETIC_OPERATION_BITWISE_OR,
|
|
ARITHMETIC_OPERATION_NO_CHECK,
|
|
ARITHMETIC_OPERATION_NO_CHECK,
|
|
TEXT_ARITHMETIC_OPERATION(|),);
|
|
break;
|
|
case GRN_OP_BITWISE_XOR :
|
|
ARITHMETIC_BINARY_OPERATION_DISPATCH(
|
|
"^",
|
|
INTEGER_ARITHMETIC_OPERATION_BITWISE_XOR,
|
|
INTEGER_ARITHMETIC_OPERATION_BITWISE_XOR,
|
|
INTEGER_ARITHMETIC_OPERATION_BITWISE_XOR,
|
|
INTEGER_ARITHMETIC_OPERATION_BITWISE_XOR,
|
|
FLOAT_ARITHMETIC_OPERATION_BITWISE_XOR,
|
|
ARITHMETIC_OPERATION_NO_CHECK,
|
|
ARITHMETIC_OPERATION_NO_CHECK,
|
|
TEXT_ARITHMETIC_OPERATION(^),);
|
|
break;
|
|
case GRN_OP_BITWISE_AND :
|
|
ARITHMETIC_BINARY_OPERATION_DISPATCH(
|
|
"&",
|
|
INTEGER_ARITHMETIC_OPERATION_BITWISE_AND,
|
|
INTEGER_ARITHMETIC_OPERATION_BITWISE_AND,
|
|
INTEGER_ARITHMETIC_OPERATION_BITWISE_AND,
|
|
INTEGER_ARITHMETIC_OPERATION_BITWISE_AND,
|
|
FLOAT_ARITHMETIC_OPERATION_BITWISE_AND,
|
|
ARITHMETIC_OPERATION_NO_CHECK,
|
|
ARITHMETIC_OPERATION_NO_CHECK,
|
|
TEXT_ARITHMETIC_OPERATION(&),);
|
|
break;
|
|
case GRN_OP_SHIFTL :
|
|
ARITHMETIC_BINARY_OPERATION_DISPATCH(
|
|
"<<",
|
|
INTEGER_ARITHMETIC_OPERATION_SHIFTL,
|
|
INTEGER_ARITHMETIC_OPERATION_SHIFTL,
|
|
INTEGER_ARITHMETIC_OPERATION_SHIFTL,
|
|
INTEGER_ARITHMETIC_OPERATION_SHIFTL,
|
|
FLOAT_ARITHMETIC_OPERATION_SHIFTL,
|
|
ARITHMETIC_OPERATION_NO_CHECK,
|
|
ARITHMETIC_OPERATION_NO_CHECK,
|
|
TEXT_ARITHMETIC_OPERATION(<<),);
|
|
break;
|
|
case GRN_OP_SHIFTR :
|
|
ARITHMETIC_BINARY_OPERATION_DISPATCH(
|
|
">>",
|
|
INTEGER_ARITHMETIC_OPERATION_SHIFTR,
|
|
INTEGER_ARITHMETIC_OPERATION_SHIFTR,
|
|
INTEGER_ARITHMETIC_OPERATION_SHIFTR,
|
|
INTEGER_ARITHMETIC_OPERATION_SHIFTR,
|
|
FLOAT_ARITHMETIC_OPERATION_SHIFTR,
|
|
ARITHMETIC_OPERATION_NO_CHECK,
|
|
ARITHMETIC_OPERATION_NO_CHECK,
|
|
TEXT_ARITHMETIC_OPERATION(>>),);
|
|
break;
|
|
case GRN_OP_SHIFTRR :
|
|
ARITHMETIC_BINARY_OPERATION_DISPATCH(
|
|
">>>",
|
|
INTEGER8_ARITHMETIC_OPERATION_SHIFTRR,
|
|
INTEGER16_ARITHMETIC_OPERATION_SHIFTRR,
|
|
INTEGER32_ARITHMETIC_OPERATION_SHIFTRR,
|
|
INTEGER64_ARITHMETIC_OPERATION_SHIFTRR,
|
|
FLOAT_ARITHMETIC_OPERATION_SHIFTRR,
|
|
ARITHMETIC_OPERATION_NO_CHECK,
|
|
ARITHMETIC_OPERATION_NO_CHECK,
|
|
{
|
|
long long unsigned int x_;
|
|
long long unsigned int y_;
|
|
|
|
res->header.type = GRN_BULK;
|
|
res->header.domain = GRN_DB_INT64;
|
|
|
|
GRN_INT64_SET(ctx, res, 0);
|
|
grn_obj_cast(ctx, x, res, GRN_FALSE);
|
|
x_ = GRN_INT64_VALUE(res);
|
|
|
|
GRN_INT64_SET(ctx, res, 0);
|
|
grn_obj_cast(ctx, y, res, GRN_FALSE);
|
|
y_ = GRN_INT64_VALUE(res);
|
|
|
|
GRN_INT64_SET(ctx, res, x_ >> y_);
|
|
}
|
|
,);
|
|
break;
|
|
case GRN_OP_INCR :
|
|
UNARY_OPERATE_AND_ASSIGN_DISPATCH(EXEC_OPERATE, 1, GRN_OBJ_INCR);
|
|
break;
|
|
case GRN_OP_DECR :
|
|
UNARY_OPERATE_AND_ASSIGN_DISPATCH(EXEC_OPERATE, 1, GRN_OBJ_DECR);
|
|
break;
|
|
case GRN_OP_INCR_POST :
|
|
UNARY_OPERATE_AND_ASSIGN_DISPATCH(EXEC_OPERATE_POST, 1, GRN_OBJ_INCR);
|
|
break;
|
|
case GRN_OP_DECR_POST :
|
|
UNARY_OPERATE_AND_ASSIGN_DISPATCH(EXEC_OPERATE_POST, 1, GRN_OBJ_DECR);
|
|
break;
|
|
case GRN_OP_NOT :
|
|
{
|
|
grn_obj *value;
|
|
grn_bool value_boolean;
|
|
POP1ALLOC1(value, res);
|
|
GRN_TRUEP(ctx, value, value_boolean);
|
|
grn_obj_reinit(ctx, res, GRN_DB_BOOL, 0);
|
|
GRN_BOOL_SET(ctx, res, !value_boolean);
|
|
}
|
|
code++;
|
|
break;
|
|
case GRN_OP_GET_MEMBER :
|
|
{
|
|
grn_obj *column_and_record_id, *index;
|
|
POP2ALLOC1(column_and_record_id, index, res);
|
|
grn_expr_exec_get_member(ctx, expr, column_and_record_id, index, res);
|
|
code++;
|
|
}
|
|
break;
|
|
default :
|
|
ERR(GRN_FUNCTION_NOT_IMPLEMENTED, "not implemented operator assigned");
|
|
goto exit;
|
|
break;
|
|
}
|
|
}
|
|
ctx->impl->stack_curr = sp - s_;
|
|
}
|
|
if (ctx->impl->stack_curr + nargs > stack_curr) {
|
|
val = grn_ctx_pop(ctx);
|
|
}
|
|
exit :
|
|
if (ctx->impl->stack_curr + nargs > stack_curr) {
|
|
/*
|
|
GRN_LOG(ctx, GRN_LOG_WARNING, "nargs=%d stack balance=%d",
|
|
nargs, stack_curr - ctx->impl->stack_curr);
|
|
*/
|
|
ctx->impl->stack_curr = stack_curr - nargs;
|
|
}
|
|
GRN_API_RETURN(val);
|
|
}
|
|
|
|
grn_obj *
|
|
grn_expr_get_value(grn_ctx *ctx, grn_obj *expr, int offset)
|
|
{
|
|
grn_obj *res = NULL;
|
|
grn_expr *e = (grn_expr *)expr;
|
|
GRN_API_ENTER;
|
|
if (0 <= offset && offset < e->values_size) {
|
|
res = &e->values[offset];
|
|
}
|
|
GRN_API_RETURN(res);
|
|
}
|
|
|
|
#define DEFAULT_WEIGHT 5
|
|
#define DEFAULT_DECAYSTEP 2
|
|
#define DEFAULT_MAX_INTERVAL 10
|
|
#define DEFAULT_SIMILARITY_THRESHOLD 0
|
|
#define DEFAULT_TERM_EXTRACT_POLICY 0
|
|
#define DEFAULT_WEIGHT_VECTOR_SIZE 4096
|
|
|
|
#define GRN_SCAN_INFO_MAX_N_ARGS 128
|
|
|
|
struct _grn_scan_info {
|
|
uint32_t start;
|
|
uint32_t end;
|
|
int32_t nargs;
|
|
int flags;
|
|
grn_operator op;
|
|
grn_operator logical_op;
|
|
grn_obj wv;
|
|
grn_obj index;
|
|
grn_obj *query;
|
|
grn_obj *args[GRN_SCAN_INFO_MAX_N_ARGS];
|
|
int max_interval;
|
|
int similarity_threshold;
|
|
grn_obj *scorer;
|
|
};
|
|
|
|
#define SI_FREE(si) do {\
|
|
GRN_OBJ_FIN(ctx, &(si)->wv);\
|
|
GRN_OBJ_FIN(ctx, &(si)->index);\
|
|
GRN_FREE(si);\
|
|
} while (0)
|
|
|
|
#define SI_ALLOC(si, i, st) do {\
|
|
if (!((si) = GRN_MALLOCN(scan_info, 1))) {\
|
|
int j;\
|
|
for (j = 0; j < i; j++) { SI_FREE(sis[j]); }\
|
|
GRN_FREE(sis);\
|
|
return NULL;\
|
|
}\
|
|
GRN_INT32_INIT(&(si)->wv, GRN_OBJ_VECTOR);\
|
|
GRN_PTR_INIT(&(si)->index, GRN_OBJ_VECTOR, GRN_ID_NIL);\
|
|
(si)->logical_op = GRN_OP_OR;\
|
|
(si)->flags = SCAN_PUSH;\
|
|
(si)->nargs = 0;\
|
|
(si)->max_interval = DEFAULT_MAX_INTERVAL;\
|
|
(si)->similarity_threshold = DEFAULT_SIMILARITY_THRESHOLD;\
|
|
(si)->start = (st);\
|
|
(si)->scorer = NULL;\
|
|
} while (0)
|
|
|
|
static scan_info **
|
|
put_logical_op(grn_ctx *ctx, scan_info **sis, int *ip, grn_operator op, int start)
|
|
{
|
|
int nparens = 1, ndifops = 0, i = *ip, j = i, r = 0;
|
|
while (j--) {
|
|
scan_info *s_ = sis[j];
|
|
if (s_->flags & SCAN_POP) {
|
|
ndifops++;
|
|
nparens++;
|
|
} else {
|
|
if (s_->flags & SCAN_PUSH) {
|
|
if (!(--nparens)) {
|
|
if (!r) {
|
|
if (ndifops) {
|
|
if (j && op != GRN_OP_AND_NOT) {
|
|
nparens = 1;
|
|
ndifops = 0;
|
|
r = j;
|
|
} else {
|
|
SI_ALLOC(s_, i, start);
|
|
s_->flags = SCAN_POP;
|
|
s_->logical_op = op;
|
|
sis[i++] = s_;
|
|
*ip = i;
|
|
break;
|
|
}
|
|
} else {
|
|
s_->flags &= ~SCAN_PUSH;
|
|
s_->logical_op = op;
|
|
break;
|
|
}
|
|
} else {
|
|
if (ndifops) {
|
|
SI_ALLOC(s_, i, start);
|
|
s_->flags = SCAN_POP;
|
|
s_->logical_op = op;
|
|
sis[i++] = s_;
|
|
*ip = i;
|
|
} else {
|
|
s_->flags &= ~SCAN_PUSH;
|
|
s_->logical_op = op;
|
|
memcpy(&sis[i], &sis[j], sizeof(scan_info *) * (r - j));
|
|
memmove(&sis[j], &sis[r], sizeof(scan_info *) * (i - r));
|
|
memcpy(&sis[i + j - r], &sis[i], sizeof(scan_info *) * (r - j));
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
} else {
|
|
if ((op == GRN_OP_AND_NOT) || (op != s_->logical_op)) {
|
|
ndifops++;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
if (j < 0) {
|
|
ERR(GRN_INVALID_ARGUMENT, "unmatched nesting level");
|
|
for (j = 0; j < i; j++) { SI_FREE(sis[j]); }
|
|
GRN_FREE(sis);
|
|
return NULL;
|
|
}
|
|
return sis;
|
|
}
|
|
|
|
/* TODO: Remove me if nobody doesn't want to reuse the implementation again. */
|
|
#if 0
|
|
static const char *opstrs[] = {
|
|
"PUSH",
|
|
"POP",
|
|
"NOP",
|
|
"CALL",
|
|
"INTERN",
|
|
"GET_REF",
|
|
"GET_VALUE",
|
|
"AND",
|
|
"AND_NOT",
|
|
"OR",
|
|
"ASSIGN",
|
|
"STAR_ASSIGN",
|
|
"SLASH_ASSIGN",
|
|
"MOD_ASSIGN",
|
|
"PLUS_ASSIGN",
|
|
"MINUS_ASSIGN",
|
|
"SHIFTL_ASSIGN",
|
|
"SHIFTR_ASSIGN",
|
|
"SHIFTRR_ASSIGN",
|
|
"AND_ASSIGN",
|
|
"XOR_ASSIGN",
|
|
"OR_ASSIGN",
|
|
"JUMP",
|
|
"CJUMP",
|
|
"COMMA",
|
|
"BITWISE_OR",
|
|
"BITWISE_XOR",
|
|
"BITWISE_AND",
|
|
"BITWISE_NOT",
|
|
"EQUAL",
|
|
"NOT_EQUAL",
|
|
"LESS",
|
|
"GREATER",
|
|
"LESS_EQUAL",
|
|
"GREATER_EQUAL",
|
|
"IN",
|
|
"MATCH",
|
|
"NEAR",
|
|
"NEAR2",
|
|
"SIMILAR",
|
|
"TERM_EXTRACT",
|
|
"SHIFTL",
|
|
"SHIFTR",
|
|
"SHIFTRR",
|
|
"PLUS",
|
|
"MINUS",
|
|
"STAR",
|
|
"SLASH",
|
|
"MOD",
|
|
"DELETE",
|
|
"INCR",
|
|
"DECR",
|
|
"INCR_POST",
|
|
"DECR_POST",
|
|
"NOT",
|
|
"ADJUST",
|
|
"EXACT",
|
|
"LCP",
|
|
"PARTIAL",
|
|
"UNSPLIT",
|
|
"PREFIX",
|
|
"SUFFIX",
|
|
"GEO_DISTANCE1",
|
|
"GEO_DISTANCE2",
|
|
"GEO_DISTANCE3",
|
|
"GEO_DISTANCE4",
|
|
"GEO_WITHINP5",
|
|
"GEO_WITHINP6",
|
|
"GEO_WITHINP8",
|
|
"OBJ_SEARCH",
|
|
"EXPR_GET_VAR",
|
|
"TABLE_CREATE",
|
|
"TABLE_SELECT",
|
|
"TABLE_SORT",
|
|
"TABLE_GROUP",
|
|
"JSON_PUT"
|
|
};
|
|
|
|
static void
|
|
put_value(grn_ctx *ctx, grn_obj *buf, grn_obj *obj)
|
|
{
|
|
int len;
|
|
char namebuf[GRN_TABLE_MAX_KEY_SIZE];
|
|
if ((len = grn_column_name(ctx, obj, namebuf, GRN_TABLE_MAX_KEY_SIZE))) {
|
|
GRN_TEXT_PUT(ctx, buf, namebuf, len);
|
|
} else {
|
|
grn_text_otoj(ctx, buf, obj, NULL);
|
|
}
|
|
}
|
|
|
|
static grn_rc
|
|
grn_expr_inspect_internal(grn_ctx *ctx, grn_obj *buf, grn_obj *expr)
|
|
{
|
|
uint32_t i, j;
|
|
grn_expr_var *var;
|
|
grn_expr_code *code;
|
|
grn_expr *e = (grn_expr *)expr;
|
|
grn_hash *vars = grn_expr_get_vars(ctx, expr, &i);
|
|
GRN_TEXT_PUTS(ctx, buf, "noname");
|
|
GRN_TEXT_PUTC(ctx, buf, '(');
|
|
{
|
|
int i = 0;
|
|
grn_obj *value;
|
|
const char *name;
|
|
uint32_t name_len;
|
|
GRN_HASH_EACH(ctx, vars, id, &name, &name_len, &value, {
|
|
if (i++) { GRN_TEXT_PUTC(ctx, buf, ','); }
|
|
GRN_TEXT_PUT(ctx, buf, name, name_len);
|
|
GRN_TEXT_PUTC(ctx, buf, ':');
|
|
put_value(ctx, buf, value);
|
|
});
|
|
}
|
|
GRN_TEXT_PUTC(ctx, buf, ')');
|
|
GRN_TEXT_PUTC(ctx, buf, '{');
|
|
for (j = 0, code = e->codes; j < e->codes_curr; j++, code++) {
|
|
if (j) { GRN_TEXT_PUTC(ctx, buf, ','); }
|
|
grn_text_itoa(ctx, buf, code->modify);
|
|
if (code->op == GRN_OP_PUSH) {
|
|
for (i = 0, var = e->vars; i < e->nvars; i++, var++) {
|
|
if (&var->value == code->value) {
|
|
GRN_TEXT_PUTC(ctx, buf, '?');
|
|
if (var->name_size) {
|
|
GRN_TEXT_PUT(ctx, buf, var->name, var->name_size);
|
|
} else {
|
|
grn_text_itoa(ctx, buf, (int)i);
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
if (i == e->nvars) {
|
|
put_value(ctx, buf, code->value);
|
|
}
|
|
} else {
|
|
if (code->value) {
|
|
put_value(ctx, buf, code->value);
|
|
GRN_TEXT_PUTC(ctx, buf, ' ');
|
|
}
|
|
GRN_TEXT_PUTS(ctx, buf, opstrs[code->op]);
|
|
}
|
|
}
|
|
GRN_TEXT_PUTC(ctx, buf, '}');
|
|
return GRN_SUCCESS;
|
|
}
|
|
|
|
#define EXPRLOG(name,expr) do {\
|
|
grn_obj strbuf;\
|
|
GRN_TEXT_INIT(&strbuf, 0);\
|
|
grn_expr_inspect_internal(ctx, &strbuf, (expr));\
|
|
GRN_TEXT_PUTC(ctx, &strbuf, '\0');\
|
|
GRN_LOG(ctx, GRN_LOG_NOTICE, "%s=(%s)", (name), GRN_TEXT_VALUE(&strbuf));\
|
|
GRN_OBJ_FIN(ctx, &strbuf);\
|
|
} while (0)
|
|
#endif
|
|
|
|
|
|
static void
|
|
scan_info_put_index(grn_ctx *ctx, scan_info *si, grn_obj *index, uint32_t sid, int32_t weight)
|
|
{
|
|
GRN_PTR_PUT(ctx, &si->index, index);
|
|
GRN_UINT32_PUT(ctx, &si->wv, sid);
|
|
GRN_INT32_PUT(ctx, &si->wv, weight);
|
|
{
|
|
int i, ni = (GRN_BULK_VSIZE(&si->index) / sizeof(grn_obj *)) - 1;
|
|
grn_obj **pi = &GRN_PTR_VALUE_AT(&si->index, ni);
|
|
for (i = 0; i < ni; i++, pi--) {
|
|
if (index == pi[-1]) {
|
|
if (i) {
|
|
int32_t *pw = &GRN_INT32_VALUE_AT(&si->wv, (ni - i) * 2);
|
|
memmove(pw + 2, pw, sizeof(int32_t) * 2 * i);
|
|
pw[0] = (int32_t) sid;
|
|
pw[1] = weight;
|
|
memmove(pi + 1, pi, sizeof(grn_obj *) * i);
|
|
pi[0] = index;
|
|
}
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
static int32_t
|
|
get_weight(grn_ctx *ctx, grn_expr_code *ec)
|
|
{
|
|
if (ec->modify == 2 && ec[2].op == GRN_OP_STAR &&
|
|
ec[1].value && ec[1].value->header.type == GRN_BULK) {
|
|
if (ec[1].value->header.domain == GRN_DB_INT32 ||
|
|
ec[1].value->header.domain == GRN_DB_UINT32) {
|
|
return GRN_INT32_VALUE(ec[1].value);
|
|
} else {
|
|
int32_t weight = 1;
|
|
grn_obj weight_buffer;
|
|
GRN_INT32_INIT(&weight_buffer, 0);
|
|
if (!grn_obj_cast(ctx, ec[1].value, &weight_buffer, GRN_FALSE)) {
|
|
weight = GRN_INT32_VALUE(&weight_buffer);
|
|
}
|
|
grn_obj_unlink(ctx, &weight_buffer);
|
|
return weight;
|
|
}
|
|
} else {
|
|
return 1;
|
|
}
|
|
}
|
|
|
|
scan_info *
|
|
grn_scan_info_open(grn_ctx *ctx, int start)
|
|
{
|
|
scan_info *si = GRN_MALLOCN(scan_info, 1);
|
|
|
|
if (!si) {
|
|
return NULL;
|
|
}
|
|
|
|
GRN_INT32_INIT(&si->wv, GRN_OBJ_VECTOR);
|
|
GRN_PTR_INIT(&si->index, GRN_OBJ_VECTOR, GRN_ID_NIL);
|
|
si->logical_op = GRN_OP_OR;
|
|
si->flags = SCAN_PUSH;
|
|
si->nargs = 0;
|
|
si->max_interval = DEFAULT_MAX_INTERVAL;
|
|
si->similarity_threshold = DEFAULT_SIMILARITY_THRESHOLD;
|
|
si->start = start;
|
|
si->scorer = NULL;
|
|
|
|
return si;
|
|
}
|
|
|
|
void
|
|
grn_scan_info_close(grn_ctx *ctx, scan_info *si)
|
|
{
|
|
SI_FREE(si);
|
|
}
|
|
|
|
void
|
|
grn_scan_info_put_index(grn_ctx *ctx, scan_info *si, grn_obj *index, uint32_t sid, int32_t weight)
|
|
{
|
|
scan_info_put_index(ctx, si, index, sid, weight);
|
|
}
|
|
|
|
scan_info **
|
|
grn_scan_info_put_logical_op(grn_ctx *ctx, scan_info **sis, int *ip,
|
|
grn_operator op, int start)
|
|
{
|
|
return put_logical_op(ctx, sis, ip, op, start);
|
|
}
|
|
|
|
int32_t
|
|
grn_expr_code_get_weight(grn_ctx *ctx, grn_expr_code *ec)
|
|
{
|
|
return get_weight(ctx, ec);
|
|
}
|
|
|
|
int
|
|
grn_scan_info_get_flags(scan_info *si)
|
|
{
|
|
return si->flags;
|
|
}
|
|
|
|
void
|
|
grn_scan_info_set_flags(scan_info *si, int flags)
|
|
{
|
|
si->flags = flags;
|
|
}
|
|
|
|
grn_operator
|
|
grn_scan_info_get_logical_op(scan_info *si)
|
|
{
|
|
return si->logical_op;
|
|
}
|
|
|
|
void
|
|
grn_scan_info_set_logical_op(scan_info *si, grn_operator logical_op)
|
|
{
|
|
si->logical_op = logical_op;
|
|
}
|
|
|
|
grn_operator
|
|
grn_scan_info_get_op(scan_info *si)
|
|
{
|
|
return si->op;
|
|
}
|
|
|
|
void
|
|
grn_scan_info_set_op(scan_info *si, grn_operator op)
|
|
{
|
|
si->op = op;
|
|
}
|
|
|
|
void
|
|
grn_scan_info_set_end(scan_info *si, uint32_t end)
|
|
{
|
|
si->end = end;
|
|
}
|
|
|
|
void
|
|
grn_scan_info_set_query(scan_info *si, grn_obj *query)
|
|
{
|
|
si->query = query;
|
|
}
|
|
|
|
int
|
|
grn_scan_info_get_max_interval(scan_info *si)
|
|
{
|
|
return si->max_interval;
|
|
}
|
|
|
|
void
|
|
grn_scan_info_set_max_interval(scan_info *si, int max_interval)
|
|
{
|
|
si->max_interval = max_interval;
|
|
}
|
|
|
|
int
|
|
grn_scan_info_get_similarity_threshold(scan_info *si)
|
|
{
|
|
return si->similarity_threshold;
|
|
}
|
|
|
|
void
|
|
grn_scan_info_set_similarity_threshold(scan_info *si, int similarity_threshold)
|
|
{
|
|
si->similarity_threshold = similarity_threshold;
|
|
}
|
|
|
|
grn_obj *
|
|
grn_scan_info_get_scorer(scan_info *si)
|
|
{
|
|
return si->scorer;
|
|
}
|
|
|
|
void
|
|
grn_scan_info_set_scorer(scan_info *si, grn_obj *scorer)
|
|
{
|
|
si->scorer = scorer;
|
|
}
|
|
|
|
grn_bool
|
|
grn_scan_info_push_arg(scan_info *si, grn_obj *arg)
|
|
{
|
|
if (si->nargs >= GRN_SCAN_INFO_MAX_N_ARGS) {
|
|
return GRN_FALSE;
|
|
}
|
|
|
|
si->args[si->nargs++] = arg;
|
|
return GRN_TRUE;
|
|
}
|
|
|
|
grn_obj *
|
|
grn_scan_info_get_arg(grn_ctx *ctx, scan_info *si, int i)
|
|
{
|
|
if (i >= si->nargs) {
|
|
return NULL;
|
|
}
|
|
return si->args[i];
|
|
}
|
|
|
|
static uint32_t
|
|
scan_info_build_find_index_column_index(grn_ctx *ctx,
|
|
scan_info *si,
|
|
grn_expr_code *ec,
|
|
uint32_t n_rest_codes,
|
|
grn_operator op)
|
|
{
|
|
uint32_t offset = 0;
|
|
grn_obj *index;
|
|
int sid = 0;
|
|
int32_t weight;
|
|
|
|
index = ec->value;
|
|
if (n_rest_codes >= 2 &&
|
|
ec[1].value &&
|
|
(ec[1].value->header.domain == GRN_DB_INT32 ||
|
|
ec[1].value->header.domain == GRN_DB_UINT32) &&
|
|
ec[2].op == GRN_OP_GET_MEMBER) {
|
|
if (ec[1].value->header.domain == GRN_DB_INT32) {
|
|
sid = GRN_INT32_VALUE(ec[1].value) + 1;
|
|
} else {
|
|
sid = GRN_UINT32_VALUE(ec[1].value) + 1;
|
|
}
|
|
offset = 2;
|
|
}
|
|
weight = get_weight(ctx, ec + offset);
|
|
scan_info_put_index(ctx, si, index, sid, weight);
|
|
|
|
return offset;
|
|
}
|
|
|
|
static uint32_t
|
|
scan_info_build_match_expr_codes(grn_ctx *ctx, scan_info *si,
|
|
grn_expr *expr, uint32_t i)
|
|
{
|
|
grn_expr_code *ec;
|
|
grn_obj *index;
|
|
int sid;
|
|
|
|
ec = &(expr->codes[i]);
|
|
if (!ec->value) {
|
|
return i + 1;
|
|
}
|
|
|
|
switch (ec->value->header.type) {
|
|
case GRN_ACCESSOR :
|
|
if (grn_column_index(ctx, ec->value, si->op, &index, 1, &sid)) {
|
|
int32_t weight = get_weight(ctx, ec);
|
|
si->flags |= SCAN_ACCESSOR;
|
|
if (((grn_accessor *)ec->value)->next) {
|
|
scan_info_put_index(ctx, si, ec->value, sid, weight);
|
|
} else {
|
|
scan_info_put_index(ctx, si, index, sid, weight);
|
|
}
|
|
}
|
|
break;
|
|
case GRN_COLUMN_FIX_SIZE :
|
|
case GRN_COLUMN_VAR_SIZE :
|
|
if (grn_column_index(ctx, ec->value, si->op, &index, 1, &sid)) {
|
|
scan_info_put_index(ctx, si, index, sid, get_weight(ctx, ec));
|
|
}
|
|
break;
|
|
case GRN_COLUMN_INDEX :
|
|
{
|
|
uint32_t n_rest_codes;
|
|
uint32_t offset;
|
|
n_rest_codes = expr->codes_curr - i;
|
|
offset = scan_info_build_find_index_column_index(ctx, si, ec,
|
|
n_rest_codes, si->op);
|
|
i += offset;
|
|
}
|
|
break;
|
|
case GRN_PROC :
|
|
if (!grn_obj_is_scorer_proc(ctx, ec->value)) {
|
|
grn_obj inspected;
|
|
GRN_TEXT_INIT(&inspected, 0);
|
|
grn_inspect(ctx, &inspected, ec->value);
|
|
ERR(GRN_INVALID_ARGUMENT,
|
|
"procedure must be scorer: <%.*s>",
|
|
(int)GRN_TEXT_LEN(&inspected),
|
|
GRN_TEXT_VALUE(&inspected));
|
|
GRN_OBJ_FIN(ctx, &inspected);
|
|
return expr->codes_curr;
|
|
}
|
|
si->scorer = ec->value;
|
|
i = scan_info_build_match_expr_codes(ctx, si, expr, i + 1);
|
|
if (expr->codes[i].op != GRN_OP_CALL) {
|
|
grn_obj inspected;
|
|
GRN_TEXT_INIT(&inspected, 0);
|
|
grn_inspect(ctx, &inspected, si->scorer);
|
|
ERR(GRN_INVALID_ARGUMENT,
|
|
"scorer must have only one argument: <%.*s>",
|
|
(int)GRN_TEXT_LEN(&inspected),
|
|
GRN_TEXT_VALUE(&inspected));
|
|
GRN_OBJ_FIN(ctx, &inspected);
|
|
return expr->codes_curr;
|
|
}
|
|
break;
|
|
case GRN_TABLE_NO_KEY :
|
|
case GRN_TABLE_HASH_KEY :
|
|
case GRN_TABLE_PAT_KEY :
|
|
case GRN_TABLE_DAT_KEY :
|
|
{
|
|
char name[GRN_TABLE_MAX_KEY_SIZE];
|
|
int name_size;
|
|
name_size = grn_obj_name(ctx, ec->value, name, GRN_TABLE_MAX_KEY_SIZE);
|
|
ERR(GRN_INVALID_ARGUMENT,
|
|
"invalid match target: <%.*s>",
|
|
name_size, name);
|
|
return expr->codes_curr;
|
|
}
|
|
break;
|
|
}
|
|
|
|
return i + 1;
|
|
}
|
|
|
|
static void
|
|
scan_info_build_match_expr(grn_ctx *ctx, scan_info *si, grn_expr *expr)
|
|
{
|
|
uint32_t i;
|
|
i = 0;
|
|
while (i < expr->codes_curr) {
|
|
i = scan_info_build_match_expr_codes(ctx, si, expr, i);
|
|
}
|
|
}
|
|
|
|
static void
|
|
scan_info_build_match(grn_ctx *ctx, scan_info *si)
|
|
{
|
|
int sid;
|
|
grn_obj *index, **p = si->args, **pe = si->args + si->nargs;
|
|
for (; p < pe; p++) {
|
|
if ((*p)->header.type == GRN_EXPR) {
|
|
scan_info_build_match_expr(ctx, si, (grn_expr *)(*p));
|
|
} else if (GRN_DB_OBJP(*p)) {
|
|
if (grn_column_index(ctx, *p, si->op, &index, 1, &sid)) {
|
|
scan_info_put_index(ctx, si, index, sid, 1);
|
|
}
|
|
} else if (GRN_ACCESSORP(*p)) {
|
|
si->flags |= SCAN_ACCESSOR;
|
|
if (grn_column_index(ctx, *p, si->op, &index, 1, &sid)) {
|
|
if (((grn_accessor *)(*p))->next) {
|
|
scan_info_put_index(ctx, si, *p, sid, 1);
|
|
} else {
|
|
scan_info_put_index(ctx, si, index, sid, 1);
|
|
}
|
|
}
|
|
} else {
|
|
switch (si->op) {
|
|
case GRN_OP_NEAR :
|
|
case GRN_OP_NEAR2 :
|
|
if (si->nargs == 3 &&
|
|
*p == si->args[2] &&
|
|
(*p)->header.domain == GRN_DB_INT32) {
|
|
si->max_interval = GRN_INT32_VALUE(*p);
|
|
} else {
|
|
si->query = *p;
|
|
}
|
|
break;
|
|
case GRN_OP_SIMILAR :
|
|
if (si->nargs == 3 &&
|
|
*p == si->args[2] &&
|
|
(*p)->header.domain == GRN_DB_INT32) {
|
|
si->similarity_threshold = GRN_INT32_VALUE(*p);
|
|
} else {
|
|
si->query = *p;
|
|
}
|
|
break;
|
|
default :
|
|
si->query = *p;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
static scan_info **
|
|
scan_info_build(grn_ctx *ctx, grn_obj *expr, int *n,
|
|
grn_operator op, uint32_t size)
|
|
{
|
|
grn_obj *var;
|
|
scan_stat stat;
|
|
int i, m = 0, o = 0;
|
|
scan_info **sis, *si = NULL;
|
|
grn_expr_code *c, *ce;
|
|
grn_expr *e = (grn_expr *)expr;
|
|
#ifdef GRN_WITH_MRUBY
|
|
if (ctx->impl->mrb.state) {
|
|
return grn_mrb_scan_info_build(ctx, expr, n, op, size);
|
|
}
|
|
#endif
|
|
if (!(var = grn_expr_get_var_by_offset(ctx, expr, 0))) { return NULL; }
|
|
for (stat = SCAN_START, c = e->codes, ce = &e->codes[e->codes_curr]; c < ce; c++) {
|
|
switch (c->op) {
|
|
case GRN_OP_MATCH :
|
|
case GRN_OP_NEAR :
|
|
case GRN_OP_NEAR2 :
|
|
case GRN_OP_SIMILAR :
|
|
case GRN_OP_PREFIX :
|
|
case GRN_OP_SUFFIX :
|
|
case GRN_OP_EQUAL :
|
|
case GRN_OP_NOT_EQUAL :
|
|
case GRN_OP_LESS :
|
|
case GRN_OP_GREATER :
|
|
case GRN_OP_LESS_EQUAL :
|
|
case GRN_OP_GREATER_EQUAL :
|
|
case GRN_OP_GEO_WITHINP5 :
|
|
case GRN_OP_GEO_WITHINP6 :
|
|
case GRN_OP_GEO_WITHINP8 :
|
|
case GRN_OP_TERM_EXTRACT :
|
|
if (stat < SCAN_COL1 || SCAN_CONST < stat) { return NULL; }
|
|
stat = SCAN_START;
|
|
m++;
|
|
break;
|
|
case GRN_OP_BITWISE_OR :
|
|
case GRN_OP_BITWISE_XOR :
|
|
case GRN_OP_BITWISE_AND :
|
|
case GRN_OP_BITWISE_NOT :
|
|
case GRN_OP_SHIFTL :
|
|
case GRN_OP_SHIFTR :
|
|
case GRN_OP_SHIFTRR :
|
|
case GRN_OP_PLUS :
|
|
case GRN_OP_MINUS :
|
|
case GRN_OP_STAR :
|
|
case GRN_OP_MOD :
|
|
if (stat < SCAN_COL1 || SCAN_CONST < stat) { return NULL; }
|
|
stat = SCAN_START;
|
|
if (m != o + 1) { return NULL; }
|
|
break;
|
|
case GRN_OP_AND :
|
|
case GRN_OP_OR :
|
|
case GRN_OP_AND_NOT :
|
|
case GRN_OP_ADJUST :
|
|
if (stat != SCAN_START) { return NULL; }
|
|
o++;
|
|
if (o >= m) { return NULL; }
|
|
break;
|
|
case GRN_OP_PUSH :
|
|
stat = (c->value == var) ? SCAN_VAR : SCAN_CONST;
|
|
break;
|
|
case GRN_OP_GET_VALUE :
|
|
switch (stat) {
|
|
case SCAN_START :
|
|
case SCAN_CONST :
|
|
case SCAN_VAR :
|
|
stat = SCAN_COL1;
|
|
break;
|
|
case SCAN_COL1 :
|
|
stat = SCAN_COL2;
|
|
break;
|
|
case SCAN_COL2 :
|
|
break;
|
|
default :
|
|
return NULL;
|
|
break;
|
|
}
|
|
break;
|
|
case GRN_OP_CALL :
|
|
if ((c->flags & GRN_EXPR_CODE_RELATIONAL_EXPRESSION) || c + 1 == ce) {
|
|
stat = SCAN_START;
|
|
m++;
|
|
} else {
|
|
stat = SCAN_COL2;
|
|
}
|
|
break;
|
|
default :
|
|
return NULL;
|
|
break;
|
|
}
|
|
}
|
|
if (stat || m != o + 1) { return NULL; }
|
|
if (!(sis = GRN_MALLOCN(scan_info *, m + m + o))) { return NULL; }
|
|
for (i = 0, stat = SCAN_START, c = e->codes, ce = &e->codes[e->codes_curr]; c < ce; c++) {
|
|
switch (c->op) {
|
|
case GRN_OP_MATCH :
|
|
case GRN_OP_NEAR :
|
|
case GRN_OP_NEAR2 :
|
|
case GRN_OP_SIMILAR :
|
|
case GRN_OP_PREFIX :
|
|
case GRN_OP_SUFFIX :
|
|
case GRN_OP_EQUAL :
|
|
case GRN_OP_NOT_EQUAL :
|
|
case GRN_OP_LESS :
|
|
case GRN_OP_GREATER :
|
|
case GRN_OP_LESS_EQUAL :
|
|
case GRN_OP_GREATER_EQUAL :
|
|
case GRN_OP_GEO_WITHINP5 :
|
|
case GRN_OP_GEO_WITHINP6 :
|
|
case GRN_OP_GEO_WITHINP8 :
|
|
case GRN_OP_TERM_EXTRACT :
|
|
stat = SCAN_START;
|
|
si->op = c->op;
|
|
si->end = c - e->codes;
|
|
sis[i++] = si;
|
|
scan_info_build_match(ctx, si);
|
|
if (ctx->rc != GRN_SUCCESS) {
|
|
int j;
|
|
for (j = 0; j < i; j++) { SI_FREE(sis[j]); }
|
|
GRN_FREE(sis);
|
|
return NULL;
|
|
}
|
|
si = NULL;
|
|
break;
|
|
case GRN_OP_AND :
|
|
case GRN_OP_OR :
|
|
case GRN_OP_AND_NOT :
|
|
case GRN_OP_ADJUST :
|
|
if (!put_logical_op(ctx, sis, &i, c->op, c - e->codes)) { return NULL; }
|
|
stat = SCAN_START;
|
|
break;
|
|
case GRN_OP_PUSH :
|
|
if (!si) { SI_ALLOC(si, i, c - e->codes); }
|
|
if (c->value == var) {
|
|
stat = SCAN_VAR;
|
|
} else {
|
|
if (si->nargs < GRN_SCAN_INFO_MAX_N_ARGS) {
|
|
si->args[si->nargs++] = c->value;
|
|
}
|
|
if (stat == SCAN_START) { si->flags |= SCAN_PRE_CONST; }
|
|
stat = SCAN_CONST;
|
|
}
|
|
break;
|
|
case GRN_OP_GET_VALUE :
|
|
switch (stat) {
|
|
case SCAN_START :
|
|
if (!si) { SI_ALLOC(si, i, c - e->codes); }
|
|
// fallthru
|
|
case SCAN_CONST :
|
|
case SCAN_VAR :
|
|
stat = SCAN_COL1;
|
|
if (si->nargs < GRN_SCAN_INFO_MAX_N_ARGS) {
|
|
si->args[si->nargs++] = c->value;
|
|
}
|
|
break;
|
|
case SCAN_COL1 :
|
|
{
|
|
int j;
|
|
grn_obj inspected;
|
|
GRN_TEXT_INIT(&inspected, 0);
|
|
GRN_TEXT_PUTS(ctx, &inspected, "<");
|
|
grn_inspect_name(ctx, &inspected, c->value);
|
|
GRN_TEXT_PUTS(ctx, &inspected, ">: <");
|
|
grn_inspect(ctx, &inspected, expr);
|
|
GRN_TEXT_PUTS(ctx, &inspected, ">");
|
|
ERR(GRN_INVALID_ARGUMENT,
|
|
"invalid expression: can't use column as a value: %.*s",
|
|
(int)GRN_TEXT_LEN(&inspected), GRN_TEXT_VALUE(&inspected));
|
|
GRN_OBJ_FIN(ctx, &inspected);
|
|
SI_FREE(si);
|
|
for (j = 0; j < i; j++) { SI_FREE(sis[j]); }
|
|
GRN_FREE(sis);
|
|
return NULL;
|
|
}
|
|
stat = SCAN_COL2;
|
|
break;
|
|
case SCAN_COL2 :
|
|
break;
|
|
default :
|
|
break;
|
|
}
|
|
break;
|
|
case GRN_OP_CALL :
|
|
if (!si) { SI_ALLOC(si, i, c - e->codes); }
|
|
if ((c->flags & GRN_EXPR_CODE_RELATIONAL_EXPRESSION) || c + 1 == ce) {
|
|
stat = SCAN_START;
|
|
si->op = c->op;
|
|
si->end = c - e->codes;
|
|
sis[i++] = si;
|
|
/* better index resolving framework for functions should be implemented */
|
|
{
|
|
int sid;
|
|
grn_obj *index, **p = si->args, **pe = si->args + si->nargs;
|
|
for (; p < pe; p++) {
|
|
if (GRN_DB_OBJP(*p)) {
|
|
if (grn_column_index(ctx, *p, c->op, &index, 1, &sid)) {
|
|
scan_info_put_index(ctx, si, index, sid, 1);
|
|
}
|
|
} else if (GRN_ACCESSORP(*p)) {
|
|
si->flags |= SCAN_ACCESSOR;
|
|
if (grn_column_index(ctx, *p, c->op, &index, 1, &sid)) {
|
|
scan_info_put_index(ctx, si, index, sid, 1);
|
|
}
|
|
} else {
|
|
si->query = *p;
|
|
}
|
|
}
|
|
}
|
|
si = NULL;
|
|
} else {
|
|
stat = SCAN_COL2;
|
|
}
|
|
break;
|
|
default :
|
|
break;
|
|
}
|
|
}
|
|
if (op == GRN_OP_OR && !size) {
|
|
// for debug
|
|
if (!(sis[0]->flags & SCAN_PUSH) || (sis[0]->logical_op != op)) {
|
|
int j;
|
|
ERR(GRN_INVALID_ARGUMENT, "invalid expr");
|
|
for (j = 0; j < i; j++) { SI_FREE(sis[j]); }
|
|
GRN_FREE(sis);
|
|
return NULL;
|
|
} else {
|
|
sis[0]->flags &= ~SCAN_PUSH;
|
|
sis[0]->logical_op = op;
|
|
}
|
|
} else {
|
|
if (!put_logical_op(ctx, sis, &i, op, c - e->codes)) { return NULL; }
|
|
}
|
|
*n = i;
|
|
return sis;
|
|
}
|
|
|
|
void
|
|
grn_inspect_scan_info_list(grn_ctx *ctx, grn_obj *buffer, scan_info **sis, int n)
|
|
{
|
|
int i;
|
|
|
|
for (i = 0; i < n; i++) {
|
|
scan_info *si = sis[i];
|
|
|
|
grn_text_printf(ctx, buffer, "[%d]\n", i);
|
|
grn_text_printf(ctx, buffer,
|
|
" op: <%s>\n",
|
|
grn_operator_to_string(si->op));
|
|
grn_text_printf(ctx, buffer,
|
|
" logical_op: <%s>\n",
|
|
grn_operator_to_string(si->logical_op));
|
|
|
|
GRN_TEXT_PUTS(ctx, buffer, " query: <");
|
|
grn_inspect(ctx, buffer, si->query);
|
|
GRN_TEXT_PUTS(ctx, buffer, ">\n");
|
|
|
|
grn_text_printf(ctx, buffer,
|
|
" expr: <%d..%d>\n", si->start, si->end);
|
|
}
|
|
}
|
|
|
|
void
|
|
grn_p_scan_info_list(grn_ctx *ctx, scan_info **sis, int n)
|
|
{
|
|
grn_obj inspected;
|
|
GRN_TEXT_INIT(&inspected, 0);
|
|
grn_inspect_scan_info_list(ctx, &inspected, sis, n);
|
|
printf("%.*s\n",
|
|
(int)GRN_TEXT_LEN(&inspected),
|
|
GRN_TEXT_VALUE(&inspected));
|
|
GRN_OBJ_FIN(ctx, &inspected);
|
|
}
|
|
|
|
inline static int32_t
|
|
exec_result_to_score(grn_ctx *ctx, grn_obj *result, grn_obj *score_buffer)
|
|
{
|
|
if (!result) {
|
|
return 0;
|
|
}
|
|
|
|
switch (result->header.type) {
|
|
case GRN_VOID :
|
|
return 0;
|
|
case GRN_BULK :
|
|
if (grn_obj_cast(ctx, result, score_buffer, GRN_FALSE) != GRN_SUCCESS) {
|
|
return 1;
|
|
}
|
|
return GRN_INT32_VALUE(score_buffer);
|
|
case GRN_UVECTOR :
|
|
case GRN_PVECTOR :
|
|
case GRN_VECTOR :
|
|
return 1;
|
|
default :
|
|
return 1; /* TODO: 1 is reasonable? */
|
|
}
|
|
}
|
|
|
|
static void
|
|
grn_table_select_sequential(grn_ctx *ctx, grn_obj *table, grn_obj *expr,
|
|
grn_obj *v, grn_obj *res, grn_operator op)
|
|
{
|
|
int32_t score;
|
|
grn_id id, *idp;
|
|
grn_table_cursor *tc;
|
|
grn_hash_cursor *hc;
|
|
grn_hash *s = (grn_hash *)res;
|
|
grn_obj *r;
|
|
grn_obj score_buffer;
|
|
GRN_RECORD_INIT(v, 0, grn_obj_id(ctx, table));
|
|
GRN_INT32_INIT(&score_buffer, 0);
|
|
switch (op) {
|
|
case GRN_OP_OR :
|
|
if ((tc = grn_table_cursor_open(ctx, table, NULL, 0, NULL, 0, 0, -1, 0))) {
|
|
while ((id = grn_table_cursor_next(ctx, tc))) {
|
|
GRN_RECORD_SET(ctx, v, id);
|
|
r = grn_expr_exec(ctx, expr, 0);
|
|
if (ctx->rc) {
|
|
break;
|
|
}
|
|
score = exec_result_to_score(ctx, r, &score_buffer);
|
|
if (score > 0) {
|
|
grn_rset_recinfo *ri;
|
|
if (grn_hash_add(ctx, s, &id, s->key_size, (void **)&ri, NULL)) {
|
|
grn_table_add_subrec(res, ri, score, (grn_rset_posinfo *)&id, 1);
|
|
}
|
|
}
|
|
}
|
|
grn_table_cursor_close(ctx, tc);
|
|
}
|
|
break;
|
|
case GRN_OP_AND :
|
|
if ((hc = grn_hash_cursor_open(ctx, s, NULL, 0, NULL, 0, 0, -1, 0))) {
|
|
while (grn_hash_cursor_next(ctx, hc)) {
|
|
grn_hash_cursor_get_key(ctx, hc, (void **) &idp);
|
|
GRN_RECORD_SET(ctx, v, *idp);
|
|
r = grn_expr_exec(ctx, expr, 0);
|
|
if (ctx->rc) {
|
|
break;
|
|
}
|
|
score = exec_result_to_score(ctx, r, &score_buffer);
|
|
if (score > 0) {
|
|
grn_rset_recinfo *ri;
|
|
grn_hash_cursor_get_value(ctx, hc, (void **) &ri);
|
|
grn_table_add_subrec(res, ri, score, (grn_rset_posinfo *)idp, 1);
|
|
} else {
|
|
grn_hash_cursor_delete(ctx, hc, NULL);
|
|
}
|
|
}
|
|
grn_hash_cursor_close(ctx, hc);
|
|
}
|
|
break;
|
|
case GRN_OP_AND_NOT :
|
|
if ((hc = grn_hash_cursor_open(ctx, s, NULL, 0, NULL, 0, 0, -1, 0))) {
|
|
while (grn_hash_cursor_next(ctx, hc)) {
|
|
grn_hash_cursor_get_key(ctx, hc, (void **) &idp);
|
|
GRN_RECORD_SET(ctx, v, *idp);
|
|
r = grn_expr_exec(ctx, expr, 0);
|
|
if (ctx->rc) {
|
|
break;
|
|
}
|
|
score = exec_result_to_score(ctx, r, &score_buffer);
|
|
if (score > 0) {
|
|
grn_hash_cursor_delete(ctx, hc, NULL);
|
|
}
|
|
}
|
|
grn_hash_cursor_close(ctx, hc);
|
|
}
|
|
break;
|
|
case GRN_OP_ADJUST :
|
|
if ((hc = grn_hash_cursor_open(ctx, s, NULL, 0, NULL, 0, 0, -1, 0))) {
|
|
while (grn_hash_cursor_next(ctx, hc)) {
|
|
grn_hash_cursor_get_key(ctx, hc, (void **) &idp);
|
|
GRN_RECORD_SET(ctx, v, *idp);
|
|
r = grn_expr_exec(ctx, expr, 0);
|
|
if (ctx->rc) {
|
|
break;
|
|
}
|
|
score = exec_result_to_score(ctx, r, &score_buffer);
|
|
if (score > 0) {
|
|
grn_rset_recinfo *ri;
|
|
grn_hash_cursor_get_value(ctx, hc, (void **) &ri);
|
|
grn_table_add_subrec(res, ri, score, (grn_rset_posinfo *)idp, 1);
|
|
}
|
|
}
|
|
grn_hash_cursor_close(ctx, hc);
|
|
}
|
|
break;
|
|
default :
|
|
break;
|
|
}
|
|
GRN_OBJ_FIN(ctx, &score_buffer);
|
|
}
|
|
|
|
static inline grn_bool
|
|
grn_table_select_index_range_column(grn_ctx *ctx, grn_obj *table,
|
|
grn_obj *index,
|
|
scan_info *si, grn_operator logical_op,
|
|
grn_obj *res)
|
|
{
|
|
grn_bool processed = GRN_FALSE;
|
|
grn_obj *index_table;
|
|
grn_obj range;
|
|
|
|
index_table = grn_ctx_at(ctx, index->header.domain);
|
|
if (!index_table) {
|
|
return GRN_FALSE;
|
|
}
|
|
|
|
GRN_OBJ_INIT(&range, GRN_BULK, 0, index_table->header.domain);
|
|
if (grn_obj_cast(ctx, si->query, &range, GRN_FALSE) == GRN_SUCCESS) {
|
|
grn_table_cursor *cursor;
|
|
const void *min = NULL, *max = NULL;
|
|
unsigned int min_size = 0, max_size = 0;
|
|
int offset = 0;
|
|
int limit = -1;
|
|
int flags = GRN_CURSOR_ASCENDING;
|
|
|
|
switch (si->op) {
|
|
case GRN_OP_LESS :
|
|
flags |= GRN_CURSOR_LT;
|
|
max = GRN_BULK_HEAD(&range);
|
|
max_size = GRN_BULK_VSIZE(&range);
|
|
break;
|
|
case GRN_OP_GREATER :
|
|
flags |= GRN_CURSOR_GT;
|
|
min = GRN_BULK_HEAD(&range);
|
|
min_size = GRN_BULK_VSIZE(&range);
|
|
break;
|
|
case GRN_OP_LESS_EQUAL :
|
|
flags |= GRN_CURSOR_LE;
|
|
max = GRN_BULK_HEAD(&range);
|
|
max_size = GRN_BULK_VSIZE(&range);
|
|
break;
|
|
case GRN_OP_GREATER_EQUAL :
|
|
flags |= GRN_CURSOR_GE;
|
|
min = GRN_BULK_HEAD(&range);
|
|
min_size = GRN_BULK_VSIZE(&range);
|
|
break;
|
|
default :
|
|
break;
|
|
}
|
|
cursor = grn_table_cursor_open(ctx, index_table,
|
|
min, min_size, max, max_size,
|
|
offset, limit, flags);
|
|
if (cursor) {
|
|
uint32_t sid;
|
|
int32_t weight;
|
|
grn_obj *index_cursor;
|
|
|
|
sid = GRN_UINT32_VALUE_AT(&(si->wv), 0);
|
|
weight = GRN_INT32_VALUE_AT(&(si->wv), 1);
|
|
index_cursor = grn_index_cursor_open(ctx, cursor, index,
|
|
GRN_ID_NIL, GRN_ID_MAX, 0);
|
|
if (index_cursor) {
|
|
grn_posting *posting;
|
|
while ((posting = grn_index_cursor_next(ctx, index_cursor, NULL))) {
|
|
if (sid == 0 || posting->sid == sid) {
|
|
grn_ii_posting ii_posting;
|
|
ii_posting.rid = posting->rid;
|
|
ii_posting.sid = posting->sid;
|
|
ii_posting.weight = posting->weight * weight;
|
|
grn_ii_posting_add(ctx, &ii_posting, (grn_hash *)res, logical_op);
|
|
}
|
|
}
|
|
processed = GRN_TRUE;
|
|
grn_obj_unlink(ctx, index_cursor);
|
|
}
|
|
grn_table_cursor_close(ctx, cursor);
|
|
}
|
|
|
|
grn_ii_resolve_sel_and(ctx, (grn_hash *)res, logical_op);
|
|
}
|
|
GRN_OBJ_FIN(ctx, &range);
|
|
|
|
grn_obj_unlink(ctx, index_table);
|
|
|
|
return processed;
|
|
}
|
|
|
|
static inline grn_bool
|
|
grn_table_select_index_range_accessor(grn_ctx *ctx, grn_obj *table,
|
|
grn_obj *accessor_stack,
|
|
scan_info *si, grn_obj *res)
|
|
{
|
|
int n_accessors;
|
|
grn_obj *current_res = NULL;
|
|
|
|
n_accessors = GRN_BULK_VSIZE(accessor_stack) / sizeof(grn_obj *);
|
|
|
|
{
|
|
grn_accessor *last_accessor;
|
|
grn_obj *target;
|
|
grn_obj *index;
|
|
grn_obj *range;
|
|
|
|
last_accessor = (grn_accessor *)GRN_PTR_VALUE_AT(accessor_stack,
|
|
n_accessors - 1);
|
|
target = last_accessor->obj;
|
|
if (grn_column_index(ctx, target, si->op, &index, 1, NULL) == 0) {
|
|
return GRN_FALSE;
|
|
}
|
|
|
|
range = grn_ctx_at(ctx, DB_OBJ(index)->range);
|
|
current_res = grn_table_create(ctx, NULL, 0, NULL,
|
|
GRN_TABLE_HASH_KEY|GRN_OBJ_WITH_SUBREC,
|
|
range,
|
|
NULL);
|
|
grn_obj_unlink(ctx, range);
|
|
if (!current_res) {
|
|
return GRN_FALSE;
|
|
}
|
|
if (!grn_table_select_index_range_column(ctx, table, index, si, GRN_OP_OR,
|
|
current_res)) {
|
|
grn_obj_unlink(ctx, current_res);
|
|
return GRN_FALSE;
|
|
}
|
|
}
|
|
|
|
{
|
|
int i;
|
|
grn_obj weight_vector;
|
|
grn_search_optarg optarg;
|
|
|
|
GRN_INT32_INIT(&weight_vector, GRN_OBJ_VECTOR);
|
|
memset(&optarg, 0, sizeof(grn_search_optarg));
|
|
if (si->op == GRN_OP_MATCH) {
|
|
optarg.mode = GRN_OP_EXACT;
|
|
} else {
|
|
optarg.mode = si->op;
|
|
}
|
|
for (i = n_accessors - 1; i > 0; i--) {
|
|
grn_rc rc = GRN_SUCCESS;
|
|
grn_accessor *accessor;
|
|
grn_obj *index;
|
|
int section;
|
|
grn_obj *domain;
|
|
grn_obj *target;
|
|
grn_obj *next_res;
|
|
grn_operator next_op;
|
|
grn_id *next_record_id = NULL;
|
|
|
|
accessor = (grn_accessor *)GRN_PTR_VALUE_AT(accessor_stack, i - 1);
|
|
target = accessor->obj;
|
|
if (grn_column_index(ctx, target, GRN_OP_EQUAL, &index, 1, §ion) == 0) {
|
|
grn_obj_unlink(ctx, current_res);
|
|
current_res = NULL;
|
|
break;
|
|
}
|
|
|
|
if (section > 0) {
|
|
int j;
|
|
int weight_position = section - 1;
|
|
|
|
GRN_BULK_REWIND(&weight_vector);
|
|
GRN_INT32_SET_AT(ctx, &weight_vector, weight_position, 1);
|
|
optarg.weight_vector = &(GRN_INT32_VALUE(&weight_vector));
|
|
optarg.vector_size = GRN_BULK_VSIZE(&weight_vector) / sizeof(int32_t);
|
|
for (j = 0; j < weight_position - 1; j++) {
|
|
optarg.weight_vector[j] = 0;
|
|
}
|
|
} else {
|
|
optarg.weight_vector = NULL;
|
|
optarg.vector_size = 1;
|
|
}
|
|
|
|
{
|
|
grn_obj *range;
|
|
range = grn_ctx_at(ctx, DB_OBJ(index)->range);
|
|
next_res = grn_table_create(ctx, NULL, 0, NULL,
|
|
GRN_TABLE_HASH_KEY|GRN_OBJ_WITH_SUBREC,
|
|
range,
|
|
NULL);
|
|
grn_obj_unlink(ctx, range);
|
|
if (!next_res) {
|
|
grn_obj_unlink(ctx, current_res);
|
|
current_res = NULL;
|
|
break;
|
|
}
|
|
next_op = GRN_OP_OR;
|
|
}
|
|
|
|
domain = grn_ctx_at(ctx, index->header.domain);
|
|
GRN_HASH_EACH(ctx, (grn_hash *)current_res, id, &next_record_id,
|
|
NULL, NULL, {
|
|
if (domain->header.type == GRN_TABLE_NO_KEY) {
|
|
rc = grn_ii_sel(ctx, (grn_ii *)index,
|
|
(const char *)next_record_id, sizeof(grn_id),
|
|
(grn_hash *)next_res, next_op, &optarg);
|
|
} else {
|
|
char key[GRN_TABLE_MAX_KEY_SIZE];
|
|
int key_len;
|
|
key_len = grn_table_get_key(ctx, domain, *next_record_id,
|
|
key, GRN_TABLE_MAX_KEY_SIZE);
|
|
rc = grn_ii_sel(ctx, (grn_ii *)index, key, key_len,
|
|
(grn_hash *)next_res, next_op, &optarg);
|
|
}
|
|
if (rc != GRN_SUCCESS) {
|
|
break;
|
|
}
|
|
});
|
|
grn_obj_unlink(ctx, domain);
|
|
grn_obj_unlink(ctx, current_res);
|
|
|
|
if (rc == GRN_SUCCESS) {
|
|
if (i == 1) {
|
|
grn_table_setoperation(ctx, res, next_res, res, si->logical_op);
|
|
grn_obj_unlink(ctx, next_res);
|
|
current_res = res;
|
|
} else {
|
|
current_res = next_res;
|
|
}
|
|
} else {
|
|
if (res != next_res) {
|
|
grn_obj_unlink(ctx, next_res);
|
|
}
|
|
current_res = NULL;
|
|
break;
|
|
}
|
|
}
|
|
GRN_OBJ_FIN(ctx, &weight_vector);
|
|
}
|
|
|
|
return current_res == res;
|
|
}
|
|
|
|
static inline grn_bool
|
|
grn_table_select_index_range(grn_ctx *ctx, grn_obj *table, grn_obj *index,
|
|
scan_info *si, grn_obj *res)
|
|
{
|
|
if (si->flags & SCAN_ACCESSOR) {
|
|
grn_bool processed;
|
|
grn_accessor *accessor = (grn_accessor *)index;
|
|
grn_accessor *a;
|
|
grn_obj accessor_stack;
|
|
|
|
if (index->header.type != GRN_ACCESSOR) {
|
|
return GRN_FALSE;
|
|
}
|
|
|
|
GRN_PTR_INIT(&accessor_stack, GRN_OBJ_VECTOR, GRN_ID_NIL);
|
|
for (a = accessor; a; a = a->next) {
|
|
GRN_PTR_PUT(ctx, &accessor_stack, a);
|
|
}
|
|
processed = grn_table_select_index_range_accessor(ctx, table,
|
|
&accessor_stack,
|
|
si, res);
|
|
GRN_OBJ_FIN(ctx, &accessor_stack);
|
|
return processed;
|
|
} else {
|
|
return grn_table_select_index_range_column(ctx, table, index, si,
|
|
si->logical_op, res);
|
|
}
|
|
}
|
|
|
|
static inline grn_bool
|
|
grn_table_select_index(grn_ctx *ctx, grn_obj *table, scan_info *si,
|
|
grn_obj *res)
|
|
{
|
|
grn_bool processed = GRN_FALSE;
|
|
if (GRN_BULK_VSIZE(&si->index)) {
|
|
grn_obj *index = GRN_PTR_VALUE(&si->index);
|
|
switch (si->op) {
|
|
case GRN_OP_EQUAL :
|
|
if (GRN_BULK_VSIZE(si->query) == 0) {
|
|
/* We can't use index for empty value. */
|
|
return GRN_FALSE;
|
|
}
|
|
if (si->flags & SCAN_ACCESSOR) {
|
|
if (index->header.type == GRN_ACCESSOR &&
|
|
!((grn_accessor *)index)->next) {
|
|
grn_obj dest;
|
|
grn_accessor *a = (grn_accessor *)index;
|
|
grn_ii_posting posting;
|
|
posting.sid = 1;
|
|
posting.pos = 0;
|
|
posting.weight = 0;
|
|
switch (a->action) {
|
|
case GRN_ACCESSOR_GET_ID :
|
|
GRN_UINT32_INIT(&dest, 0);
|
|
if (!grn_obj_cast(ctx, si->query, &dest, GRN_FALSE)) {
|
|
posting.rid = GRN_UINT32_VALUE(&dest);
|
|
if (posting.rid) {
|
|
if (posting.rid == grn_table_at(ctx, table, posting.rid)) {
|
|
grn_ii_posting_add(ctx, &posting, (grn_hash *)res,
|
|
si->logical_op);
|
|
}
|
|
}
|
|
processed = GRN_TRUE;
|
|
}
|
|
grn_ii_resolve_sel_and(ctx, (grn_hash *)res, si->logical_op);
|
|
GRN_OBJ_FIN(ctx, &dest);
|
|
break;
|
|
case GRN_ACCESSOR_GET_KEY :
|
|
GRN_OBJ_INIT(&dest, GRN_BULK, 0, table->header.domain);
|
|
if (!grn_obj_cast(ctx, si->query, &dest, GRN_FALSE)) {
|
|
if ((posting.rid = grn_table_get(ctx, table,
|
|
GRN_BULK_HEAD(&dest),
|
|
GRN_BULK_VSIZE(&dest)))) {
|
|
grn_ii_posting_add(ctx, &posting, (grn_hash *)res,
|
|
si->logical_op);
|
|
}
|
|
processed = GRN_TRUE;
|
|
}
|
|
grn_ii_resolve_sel_and(ctx, (grn_hash *)res, si->logical_op);
|
|
GRN_OBJ_FIN(ctx, &dest);
|
|
break;
|
|
}
|
|
}
|
|
} else {
|
|
grn_obj *domain = grn_ctx_at(ctx, index->header.domain);
|
|
if (domain) {
|
|
grn_id tid;
|
|
if (GRN_OBJ_GET_DOMAIN(si->query) == DB_OBJ(domain)->id) {
|
|
tid = GRN_RECORD_VALUE(si->query);
|
|
} else {
|
|
tid = grn_table_get(ctx, domain,
|
|
GRN_BULK_HEAD(si->query),
|
|
GRN_BULK_VSIZE(si->query));
|
|
}
|
|
grn_ii_at(ctx, (grn_ii *)index, tid, (grn_hash *)res, si->logical_op);
|
|
}
|
|
grn_ii_resolve_sel_and(ctx, (grn_hash *)res, si->logical_op);
|
|
processed = GRN_TRUE;
|
|
}
|
|
break;
|
|
case GRN_OP_SUFFIX :
|
|
{
|
|
grn_obj *domain;
|
|
if (si->flags & SCAN_ACCESSOR) {
|
|
domain = table;
|
|
} else {
|
|
domain = grn_ctx_at(ctx, index->header.domain);
|
|
}
|
|
if (domain->header.type != GRN_TABLE_PAT_KEY) {
|
|
break;
|
|
}
|
|
if (!(domain->header.flags & GRN_OBJ_KEY_WITH_SIS)) {
|
|
break;
|
|
}
|
|
}
|
|
/* fallthru */
|
|
case GRN_OP_PREFIX :
|
|
if (si->flags & SCAN_ACCESSOR) {
|
|
if (index->header.type == GRN_ACCESSOR &&
|
|
!((grn_accessor *)index)->next) {
|
|
grn_obj dest;
|
|
grn_accessor *a = (grn_accessor *)index;
|
|
grn_ii_posting posting;
|
|
posting.sid = 1;
|
|
posting.pos = 0;
|
|
posting.weight = 0;
|
|
switch (a->action) {
|
|
case GRN_ACCESSOR_GET_ID :
|
|
/* todo */
|
|
break;
|
|
case GRN_ACCESSOR_GET_KEY :
|
|
GRN_OBJ_INIT(&dest, GRN_BULK, 0, table->header.domain);
|
|
if (!grn_obj_cast(ctx, si->query, &dest, GRN_FALSE)) {
|
|
grn_hash *pres;
|
|
if ((pres = grn_hash_create(ctx, NULL, sizeof(grn_id), 0,
|
|
GRN_OBJ_TABLE_HASH_KEY))) {
|
|
grn_id *key;
|
|
grn_table_search(ctx, table,
|
|
GRN_BULK_HEAD(&dest), GRN_BULK_VSIZE(&dest),
|
|
si->op, (grn_obj *)pres, GRN_OP_OR);
|
|
GRN_HASH_EACH(ctx, pres, id, &key, NULL, NULL, {
|
|
posting.rid = *key;
|
|
grn_ii_posting_add(ctx, &posting, (grn_hash *)res,
|
|
si->logical_op);
|
|
});
|
|
grn_hash_close(ctx, pres);
|
|
}
|
|
processed = GRN_TRUE;
|
|
}
|
|
grn_ii_resolve_sel_and(ctx, (grn_hash *)res, si->logical_op);
|
|
GRN_OBJ_FIN(ctx, &dest);
|
|
break;
|
|
}
|
|
}
|
|
} else {
|
|
grn_obj *i = GRN_PTR_VALUE(&si->index);
|
|
grn_obj *domain = grn_ctx_at(ctx, i->header.domain);
|
|
if (domain) {
|
|
grn_hash *pres;
|
|
if ((pres = grn_hash_create(ctx, NULL, sizeof(grn_id), 0,
|
|
GRN_OBJ_TABLE_HASH_KEY))) {
|
|
grn_id *key;
|
|
grn_table_search(ctx, domain,
|
|
GRN_BULK_HEAD(si->query),
|
|
GRN_BULK_VSIZE(si->query),
|
|
si->op, (grn_obj *)pres, GRN_OP_OR);
|
|
grn_obj_unlink(ctx, domain);
|
|
GRN_HASH_EACH(ctx, pres, id, &key, NULL, NULL, {
|
|
grn_ii_at(ctx, (grn_ii *)index, *key, (grn_hash *)res, si->logical_op);
|
|
});
|
|
grn_hash_close(ctx, pres);
|
|
}
|
|
grn_obj_unlink(ctx, domain);
|
|
}
|
|
grn_ii_resolve_sel_and(ctx, (grn_hash *)res, si->logical_op);
|
|
processed = GRN_TRUE;
|
|
}
|
|
break;
|
|
case GRN_OP_MATCH :
|
|
case GRN_OP_NEAR :
|
|
case GRN_OP_NEAR2 :
|
|
case GRN_OP_SIMILAR :
|
|
{
|
|
grn_obj wv, **ip = &GRN_PTR_VALUE(&si->index);
|
|
int j = GRN_BULK_VSIZE(&si->index)/sizeof(grn_obj *);
|
|
int32_t *wp = &GRN_INT32_VALUE(&si->wv);
|
|
grn_search_optarg optarg;
|
|
GRN_INT32_INIT(&wv, GRN_OBJ_VECTOR);
|
|
if (si->op == GRN_OP_MATCH) {
|
|
optarg.mode = GRN_OP_EXACT;
|
|
} else {
|
|
optarg.mode = si->op;
|
|
}
|
|
optarg.max_interval = 0;
|
|
optarg.similarity_threshold = 0;
|
|
switch (si->op) {
|
|
case GRN_OP_NEAR :
|
|
case GRN_OP_NEAR2 :
|
|
optarg.max_interval = si->max_interval;
|
|
break;
|
|
case GRN_OP_SIMILAR :
|
|
optarg.similarity_threshold = si->similarity_threshold;
|
|
break;
|
|
default :
|
|
break;
|
|
}
|
|
optarg.weight_vector = (int *)GRN_BULK_HEAD(&wv);
|
|
/* optarg.vector_size = GRN_BULK_VSIZE(&si->wv); */
|
|
optarg.vector_size = 1;
|
|
optarg.proc = NULL;
|
|
optarg.max_size = 0;
|
|
optarg.scorer = si->scorer;
|
|
ctx->flags |= GRN_CTX_TEMPORARY_DISABLE_II_RESOLVE_SEL_AND;
|
|
for (; j--; ip++, wp += 2) {
|
|
uint32_t sid = (uint32_t) wp[0];
|
|
int32_t weight = wp[1];
|
|
if (sid) {
|
|
int weight_index = sid - 1;
|
|
GRN_INT32_SET_AT(ctx, &wv, weight_index, weight);
|
|
optarg.weight_vector = &GRN_INT32_VALUE(&wv);
|
|
optarg.vector_size = GRN_BULK_VSIZE(&wv)/sizeof(int32_t);
|
|
} else {
|
|
optarg.weight_vector = NULL;
|
|
optarg.vector_size = weight;
|
|
}
|
|
if (j) {
|
|
if (sid && ip[0] == ip[1]) { continue; }
|
|
} else {
|
|
ctx->flags &= ~GRN_CTX_TEMPORARY_DISABLE_II_RESOLVE_SEL_AND;
|
|
}
|
|
grn_obj_search(ctx, ip[0], si->query, res, si->logical_op, &optarg);
|
|
if (optarg.weight_vector) {
|
|
int i;
|
|
for (i = 0; i < optarg.vector_size; i++) {
|
|
optarg.weight_vector[i] = 0;
|
|
}
|
|
}
|
|
GRN_BULK_REWIND(&wv);
|
|
}
|
|
GRN_OBJ_FIN(ctx, &wv);
|
|
}
|
|
processed = GRN_TRUE;
|
|
break;
|
|
case GRN_OP_TERM_EXTRACT :
|
|
if (si->flags & SCAN_ACCESSOR) {
|
|
if (index->header.type == GRN_ACCESSOR &&
|
|
!((grn_accessor *)index)->next) {
|
|
grn_accessor *a = (grn_accessor *)index;
|
|
switch (a->action) {
|
|
case GRN_ACCESSOR_GET_KEY :
|
|
grn_table_search(ctx, table,
|
|
GRN_TEXT_VALUE(si->query), GRN_TEXT_LEN(si->query),
|
|
GRN_OP_TERM_EXTRACT, res, si->logical_op);
|
|
processed = GRN_TRUE;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
case GRN_OP_CALL :
|
|
if (grn_obj_is_selector_proc(ctx, si->args[0])) {
|
|
grn_rc rc;
|
|
grn_proc *proc = (grn_proc *)(si->args[0]);
|
|
rc = proc->selector(ctx, table, index, si->nargs, si->args,
|
|
res, si->logical_op);
|
|
if (rc) {
|
|
/* TODO: report error */
|
|
} else {
|
|
processed = GRN_TRUE;
|
|
}
|
|
}
|
|
break;
|
|
case GRN_OP_LESS :
|
|
case GRN_OP_GREATER :
|
|
case GRN_OP_LESS_EQUAL :
|
|
case GRN_OP_GREATER_EQUAL :
|
|
processed = grn_table_select_index_range(ctx, table, index, si, res);
|
|
break;
|
|
default :
|
|
/* todo : implement */
|
|
/* todo : handle SCAN_PRE_CONST */
|
|
break;
|
|
}
|
|
} else {
|
|
switch (si->op) {
|
|
case GRN_OP_CALL :
|
|
if (grn_obj_is_selector_proc(ctx, si->args[0])) {
|
|
grn_rc rc;
|
|
grn_proc *proc = (grn_proc *)(si->args[0]);
|
|
rc = proc->selector(ctx, table, NULL, si->nargs, si->args,
|
|
res, si->logical_op);
|
|
if (rc) {
|
|
/* TODO: report error */
|
|
} else {
|
|
processed = GRN_TRUE;
|
|
}
|
|
}
|
|
default :
|
|
break;
|
|
}
|
|
}
|
|
return processed;
|
|
}
|
|
|
|
grn_obj *
|
|
grn_table_select(grn_ctx *ctx, grn_obj *table, grn_obj *expr,
|
|
grn_obj *res, grn_operator op)
|
|
{
|
|
grn_obj *v;
|
|
unsigned int res_size;
|
|
grn_bool res_created = GRN_FALSE;
|
|
if (res) {
|
|
if (res->header.type != GRN_TABLE_HASH_KEY ||
|
|
(res->header.domain != DB_OBJ(table)->id)) {
|
|
ERR(GRN_INVALID_ARGUMENT, "hash table required");
|
|
return NULL;
|
|
}
|
|
} else {
|
|
if (!(res = grn_table_create(ctx, NULL, 0, NULL,
|
|
GRN_TABLE_HASH_KEY|GRN_OBJ_WITH_SUBREC, table, NULL))) {
|
|
return NULL;
|
|
}
|
|
res_created = GRN_TRUE;
|
|
}
|
|
if (!(v = grn_expr_get_var_by_offset(ctx, expr, 0))) {
|
|
ERR(GRN_INVALID_ARGUMENT, "at least one variable must be defined");
|
|
return NULL;
|
|
}
|
|
GRN_API_ENTER;
|
|
res_size = GRN_HASH_SIZE((grn_hash *)res);
|
|
if (op == GRN_OP_OR || res_size) {
|
|
int i, n;
|
|
scan_info **sis;
|
|
if ((sis = scan_info_build(ctx, expr, &n, op, res_size))) {
|
|
grn_obj res_stack;
|
|
grn_expr *e = (grn_expr *)expr;
|
|
grn_expr_code *codes = e->codes;
|
|
uint32_t codes_curr = e->codes_curr;
|
|
GRN_PTR_INIT(&res_stack, GRN_OBJ_VECTOR, GRN_ID_NIL);
|
|
for (i = 0; i < n; i++) {
|
|
scan_info *si = sis[i];
|
|
if (si->flags & SCAN_POP) {
|
|
grn_obj *res_;
|
|
GRN_PTR_POP(&res_stack, res_);
|
|
grn_table_setoperation(ctx, res_, res, res_, si->logical_op);
|
|
grn_obj_close(ctx, res);
|
|
res = res_;
|
|
} else {
|
|
grn_bool processed = GRN_FALSE;
|
|
if (si->flags & SCAN_PUSH) {
|
|
grn_obj *res_ = NULL;
|
|
res_ = grn_table_create(ctx, NULL, 0, NULL,
|
|
GRN_TABLE_HASH_KEY|GRN_OBJ_WITH_SUBREC, table, NULL);
|
|
if (!res_) {
|
|
int i = 0;
|
|
if (!res_created) { i++; }
|
|
for (; i < GRN_BULK_VSIZE(&res_stack) / sizeof(grn_obj *); i++) {
|
|
grn_obj *stacked_res;
|
|
stacked_res = *((grn_obj **)GRN_BULK_HEAD(&res_stack) + i);
|
|
grn_obj_close(ctx, stacked_res);
|
|
}
|
|
break;
|
|
}
|
|
GRN_PTR_PUT(ctx, &res_stack, res);
|
|
res = res_;
|
|
}
|
|
processed = grn_table_select_index(ctx, table, si, res);
|
|
if (!processed) {
|
|
if (ctx->rc) { break; }
|
|
e->codes = codes + si->start;
|
|
e->codes_curr = si->end - si->start + 1;
|
|
grn_table_select_sequential(ctx, table, expr, v,
|
|
res, si->logical_op);
|
|
}
|
|
}
|
|
GRN_QUERY_LOG(ctx, GRN_QUERY_LOG_SIZE,
|
|
":", "filter(%d)", grn_table_size(ctx, res));
|
|
if (ctx->rc) { break; }
|
|
}
|
|
for (i = 0; i < n; i++) {
|
|
scan_info *si = sis[i];
|
|
SI_FREE(si);
|
|
}
|
|
GRN_OBJ_FIN(ctx, &res_stack);
|
|
GRN_FREE(sis);
|
|
e->codes = codes;
|
|
e->codes_curr = codes_curr;
|
|
} else {
|
|
if (!ctx->rc) {
|
|
grn_table_select_sequential(ctx, table, expr, v, res, op);
|
|
}
|
|
}
|
|
}
|
|
GRN_API_RETURN(res);
|
|
}
|
|
|
|
/* grn_expr_parse */
|
|
|
|
grn_obj *
|
|
grn_ptr_value_at(grn_obj *obj, int offset)
|
|
{
|
|
int size = GRN_BULK_VSIZE(obj) / sizeof(grn_obj *);
|
|
if (offset < 0) { offset = size + offset; }
|
|
return (0 <= offset && offset < size)
|
|
? (((grn_obj **)GRN_BULK_HEAD(obj))[offset])
|
|
: NULL;
|
|
}
|
|
|
|
int32_t
|
|
grn_int32_value_at(grn_obj *obj, int offset)
|
|
{
|
|
int size = GRN_BULK_VSIZE(obj) / sizeof(int32_t);
|
|
if (offset < 0) { offset = size + offset; }
|
|
return (0 <= offset && offset < size)
|
|
? (((int32_t *)GRN_BULK_HEAD(obj))[offset])
|
|
: 0;
|
|
}
|
|
|
|
/* grn_expr_create_from_str */
|
|
|
|
#include "grn_snip.h"
|
|
|
|
typedef struct {
|
|
grn_ctx *ctx;
|
|
grn_obj *e;
|
|
grn_obj *v;
|
|
const char *str;
|
|
const char *cur;
|
|
const char *str_end;
|
|
grn_obj *table;
|
|
grn_obj *default_column;
|
|
grn_obj buf;
|
|
grn_obj token_stack;
|
|
grn_obj column_stack;
|
|
grn_obj op_stack;
|
|
grn_obj mode_stack;
|
|
grn_obj max_interval_stack;
|
|
grn_obj similarity_threshold_stack;
|
|
grn_operator default_op;
|
|
grn_select_optarg opt;
|
|
grn_operator default_mode;
|
|
grn_expr_flags flags;
|
|
grn_expr_flags default_flags;
|
|
int escalation_threshold;
|
|
int escalation_decaystep;
|
|
int weight_offset;
|
|
grn_hash *weight_set;
|
|
snip_cond *snip_conds;
|
|
} efs_info;
|
|
|
|
typedef struct {
|
|
grn_operator op;
|
|
int weight;
|
|
} efs_op;
|
|
|
|
inline static void
|
|
skip_space(grn_ctx *ctx, efs_info *q)
|
|
{
|
|
unsigned int len;
|
|
while (q->cur < q->str_end && grn_isspace(q->cur, ctx->encoding)) {
|
|
/* null check and length check */
|
|
if (!(len = grn_charlen(ctx, q->cur, q->str_end))) {
|
|
q->cur = q->str_end;
|
|
break;
|
|
}
|
|
q->cur += len;
|
|
}
|
|
}
|
|
|
|
static grn_rc get_expr(grn_ctx *ctx, efs_info *q, grn_obj *column, grn_operator mode);
|
|
static grn_rc get_token(grn_ctx *ctx, efs_info *q, efs_op *op, grn_obj *column, grn_operator mode);
|
|
|
|
static grn_rc
|
|
get_phrase(grn_ctx *ctx, efs_info *q, grn_obj *column, int mode, int option)
|
|
{
|
|
const char *start, *s;
|
|
start = s = q->cur;
|
|
GRN_BULK_REWIND(&q->buf);
|
|
while (1) {
|
|
unsigned int len;
|
|
if (s >= q->str_end) {
|
|
q->cur = s;
|
|
break;
|
|
}
|
|
len = grn_charlen(ctx, s, q->str_end);
|
|
if (len == 0) {
|
|
/* invalid string containing malformed multibyte char */
|
|
return GRN_END_OF_DATA;
|
|
} else if (len == 1) {
|
|
if (*s == GRN_QUERY_QUOTER) {
|
|
q->cur = s + 1;
|
|
break;
|
|
} else if (*s == GRN_QUERY_ESCAPE && s + 1 < q->str_end) {
|
|
s++;
|
|
len = grn_charlen(ctx, s, q->str_end);
|
|
}
|
|
}
|
|
GRN_TEXT_PUT(ctx, &q->buf, s, len);
|
|
s += len;
|
|
}
|
|
grn_expr_append_obj(ctx, q->e, q->v, GRN_OP_PUSH, 1);
|
|
grn_expr_append_const(ctx, q->e, column, GRN_OP_PUSH, 1);
|
|
grn_expr_append_op(ctx, q->e, GRN_OP_GET_VALUE, 2);
|
|
grn_expr_append_const(ctx, q->e, &q->buf, GRN_OP_PUSH, 1);
|
|
if (mode == GRN_OP_MATCH || mode == GRN_OP_EXACT) {
|
|
grn_expr_append_op(ctx, q->e, mode, 2);
|
|
} else {
|
|
grn_expr_append_const_int(ctx, q->e, option, GRN_OP_PUSH, 1);
|
|
grn_expr_append_op(ctx, q->e, mode, 3);
|
|
}
|
|
return GRN_SUCCESS;
|
|
}
|
|
|
|
static grn_rc
|
|
get_geocond(grn_ctx *ctx, efs_info *q, grn_obj *longitude, grn_obj *latitude)
|
|
{
|
|
unsigned int len;
|
|
const char *start = q->cur, *end;
|
|
for (end = q->cur;; ) {
|
|
/* null check and length check */
|
|
if (!(len = grn_charlen(ctx, end, q->str_end))) {
|
|
q->cur = q->str_end;
|
|
break;
|
|
}
|
|
if (grn_isspace(end, ctx->encoding) ||
|
|
*end == GRN_QUERY_PARENR) {
|
|
q->cur = end;
|
|
break;
|
|
}
|
|
}
|
|
{
|
|
const char *tokbuf[8];
|
|
int32_t lng0, lat0, lng1, lat1, lng2, lat2, r;
|
|
int32_t n = grn_str_tok((char *)start, end - start, ',', tokbuf, 8, NULL);
|
|
switch (n) {
|
|
case 3 :
|
|
lng0 = grn_atoi(start, tokbuf[0], NULL);
|
|
lat0 = grn_atoi(tokbuf[0] + 1, tokbuf[1], NULL);
|
|
r = grn_atoi(tokbuf[1] + 1, tokbuf[2], NULL);
|
|
grn_expr_append_obj(ctx, q->e, q->v, GRN_OP_PUSH, 1);
|
|
grn_expr_append_const(ctx, q->e, longitude, GRN_OP_PUSH, 1);
|
|
grn_expr_append_op(ctx, q->e, GRN_OP_GET_VALUE, 2);
|
|
grn_expr_append_obj(ctx, q->e, q->v, GRN_OP_PUSH, 1);
|
|
grn_expr_append_const(ctx, q->e, latitude, GRN_OP_PUSH, 1);
|
|
grn_expr_append_op(ctx, q->e, GRN_OP_GET_VALUE, 2);
|
|
grn_expr_append_const_int(ctx, q->e, lng0, GRN_OP_PUSH, 1);
|
|
grn_expr_append_const_int(ctx, q->e, lat0, GRN_OP_PUSH, 1);
|
|
grn_expr_append_const_int(ctx, q->e, r, GRN_OP_PUSH, 1);
|
|
grn_expr_append_op(ctx, q->e, GRN_OP_GEO_WITHINP5, 5);
|
|
break;
|
|
case 4 :
|
|
lng0 = grn_atoi(start, tokbuf[0], NULL);
|
|
lat0 = grn_atoi(tokbuf[0] + 1, tokbuf[1], NULL);
|
|
lng1 = grn_atoi(tokbuf[1] + 1, tokbuf[2], NULL);
|
|
lat1 = grn_atoi(tokbuf[2] + 1, tokbuf[3], NULL);
|
|
grn_expr_append_obj(ctx, q->e, q->v, GRN_OP_PUSH, 1);
|
|
grn_expr_append_const(ctx, q->e, longitude, GRN_OP_PUSH, 1);
|
|
grn_expr_append_op(ctx, q->e, GRN_OP_GET_VALUE, 2);
|
|
grn_expr_append_obj(ctx, q->e, q->v, GRN_OP_PUSH, 1);
|
|
grn_expr_append_const(ctx, q->e, latitude, GRN_OP_PUSH, 1);
|
|
grn_expr_append_op(ctx, q->e, GRN_OP_GET_VALUE, 2);
|
|
grn_expr_append_const_int(ctx, q->e, lng0, GRN_OP_PUSH, 1);
|
|
grn_expr_append_const_int(ctx, q->e, lat0, GRN_OP_PUSH, 1);
|
|
grn_expr_append_const_int(ctx, q->e, lng1, GRN_OP_PUSH, 1);
|
|
grn_expr_append_const_int(ctx, q->e, lat1, GRN_OP_PUSH, 1);
|
|
grn_expr_append_op(ctx, q->e, GRN_OP_GEO_WITHINP6, 6);
|
|
break;
|
|
case 6 :
|
|
lng0 = grn_atoi(start, tokbuf[0], NULL);
|
|
lat0 = grn_atoi(tokbuf[0] + 1, tokbuf[1], NULL);
|
|
lng1 = grn_atoi(tokbuf[1] + 1, tokbuf[2], NULL);
|
|
lat1 = grn_atoi(tokbuf[2] + 1, tokbuf[3], NULL);
|
|
lng2 = grn_atoi(tokbuf[3] + 1, tokbuf[4], NULL);
|
|
lat2 = grn_atoi(tokbuf[4] + 1, tokbuf[5], NULL);
|
|
grn_expr_append_obj(ctx, q->e, q->v, GRN_OP_PUSH, 1);
|
|
grn_expr_append_const(ctx, q->e, longitude, GRN_OP_PUSH, 1);
|
|
grn_expr_append_op(ctx, q->e, GRN_OP_GET_VALUE, 2);
|
|
grn_expr_append_obj(ctx, q->e, q->v, GRN_OP_PUSH, 1);
|
|
grn_expr_append_const(ctx, q->e, latitude, GRN_OP_PUSH, 1);
|
|
grn_expr_append_op(ctx, q->e, GRN_OP_GET_VALUE, 2);
|
|
grn_expr_append_const_int(ctx, q->e, lng0, GRN_OP_PUSH, 1);
|
|
grn_expr_append_const_int(ctx, q->e, lat0, GRN_OP_PUSH, 1);
|
|
grn_expr_append_const_int(ctx, q->e, lng1, GRN_OP_PUSH, 1);
|
|
grn_expr_append_const_int(ctx, q->e, lat1, GRN_OP_PUSH, 1);
|
|
grn_expr_append_const_int(ctx, q->e, lng2, GRN_OP_PUSH, 1);
|
|
grn_expr_append_const_int(ctx, q->e, lat2, GRN_OP_PUSH, 1);
|
|
grn_expr_append_op(ctx, q->e, GRN_OP_GEO_WITHINP8, 8);
|
|
break;
|
|
default :
|
|
ERR(GRN_INVALID_ARGUMENT, "invalid geocond");
|
|
break;
|
|
}
|
|
}
|
|
return ctx->rc;
|
|
}
|
|
|
|
static grn_rc
|
|
get_word(grn_ctx *ctx, efs_info *q, grn_obj *column, int mode, int option)
|
|
{
|
|
const char *start = q->cur, *end;
|
|
unsigned int len;
|
|
for (end = q->cur;; ) {
|
|
/* null check and length check */
|
|
if (!(len = grn_charlen(ctx, end, q->str_end))) {
|
|
q->cur = q->str_end;
|
|
break;
|
|
}
|
|
if (grn_isspace(end, ctx->encoding) ||
|
|
*end == GRN_QUERY_PARENR) {
|
|
q->cur = end;
|
|
break;
|
|
}
|
|
if (*end == GRN_QUERY_COLUMN) {
|
|
grn_obj *c = grn_obj_column(ctx, q->table, start, end - start);
|
|
if (c && end + 1 < q->str_end) {
|
|
efs_op op;
|
|
switch (end[1]) {
|
|
case '!' :
|
|
mode = GRN_OP_NOT_EQUAL;
|
|
q->cur = end + 2;
|
|
break;
|
|
case '=' :
|
|
if (q->flags & GRN_EXPR_ALLOW_UPDATE) {
|
|
mode = GRN_OP_ASSIGN;
|
|
q->cur = end + 2;
|
|
} else {
|
|
get_token(ctx, q, &op, c, mode);
|
|
}
|
|
break;
|
|
case '<' :
|
|
if (end + 2 < q->str_end && end[2] == '=') {
|
|
mode = GRN_OP_LESS_EQUAL;
|
|
q->cur = end + 3;
|
|
} else {
|
|
mode = GRN_OP_LESS;
|
|
q->cur = end + 2;
|
|
}
|
|
break;
|
|
case '>' :
|
|
if (end + 2 < q->str_end && end[2] == '=') {
|
|
mode = GRN_OP_GREATER_EQUAL;
|
|
q->cur = end + 3;
|
|
} else {
|
|
mode = GRN_OP_GREATER;
|
|
q->cur = end + 2;
|
|
}
|
|
break;
|
|
case '%' :
|
|
mode = GRN_OP_MATCH;
|
|
q->cur = end + 2;
|
|
break;
|
|
case '@' :
|
|
q->cur = end + 2;
|
|
return get_geocond(ctx, q, column, c);
|
|
break;
|
|
default :
|
|
mode = GRN_OP_EQUAL;
|
|
q->cur = end + 1;
|
|
break;
|
|
}
|
|
return get_token(ctx, q, &op, c, mode);
|
|
} else {
|
|
ERR(GRN_INVALID_ARGUMENT, "column lookup failed");
|
|
return ctx->rc;
|
|
}
|
|
} else if (*end == GRN_QUERY_PREFIX) {
|
|
mode = GRN_OP_PREFIX;
|
|
q->cur = end + 1;
|
|
break;
|
|
}
|
|
end += len;
|
|
}
|
|
if (!column) {
|
|
ERR(GRN_INVALID_ARGUMENT, "column missing");
|
|
return ctx->rc;
|
|
}
|
|
if (mode == GRN_OP_ASSIGN) {
|
|
grn_expr_append_obj(ctx, q->e, q->v, GRN_OP_PUSH, 1);
|
|
grn_expr_append_const(ctx, q->e, column, GRN_OP_PUSH, 1);
|
|
grn_expr_append_const_str(ctx, q->e, start, end - start, GRN_OP_PUSH, 1);
|
|
grn_expr_append_op(ctx, q->e, GRN_OP_ASSIGN, 2);
|
|
} else {
|
|
grn_expr_append_obj(ctx, q->e, q->v, GRN_OP_PUSH, 1);
|
|
grn_expr_append_const(ctx, q->e, column, GRN_OP_PUSH, 1);
|
|
grn_expr_append_op(ctx, q->e, GRN_OP_GET_VALUE, 2);
|
|
grn_expr_append_const_str(ctx, q->e, start, end - start, GRN_OP_PUSH, 1);
|
|
switch (mode) {
|
|
case GRN_OP_NEAR :
|
|
case GRN_OP_NEAR2 :
|
|
case GRN_OP_SIMILAR :
|
|
case GRN_OP_TERM_EXTRACT :
|
|
grn_expr_append_const_int(ctx, q->e, option, GRN_OP_PUSH, 1);
|
|
grn_expr_append_op(ctx, q->e, mode, 3);
|
|
break;
|
|
default :
|
|
grn_expr_append_op(ctx, q->e, mode, 2);
|
|
break;
|
|
}
|
|
}
|
|
return GRN_SUCCESS;
|
|
}
|
|
|
|
static grn_bool
|
|
get_op(efs_info *q, efs_op *op, grn_operator *mode, int *option)
|
|
{
|
|
grn_bool found = GRN_TRUE;
|
|
const char *start, *end = q->cur;
|
|
switch (*end) {
|
|
case 'S' :
|
|
*mode = GRN_OP_SIMILAR;
|
|
start = ++end;
|
|
*option = grn_atoi(start, q->str_end, (const char **)&end);
|
|
if (start == end) { *option = DEFAULT_SIMILARITY_THRESHOLD; }
|
|
q->cur = end;
|
|
break;
|
|
case 'N' :
|
|
*mode = GRN_OP_NEAR;
|
|
start = ++end;
|
|
*option = grn_atoi(start, q->str_end, (const char **)&end);
|
|
if (start == end) { *option = DEFAULT_MAX_INTERVAL; }
|
|
q->cur = end;
|
|
break;
|
|
case 'n' :
|
|
*mode = GRN_OP_NEAR2;
|
|
start = ++end;
|
|
*option = grn_atoi(start, q->str_end, (const char **)&end);
|
|
if (start == end) { *option = DEFAULT_MAX_INTERVAL; }
|
|
q->cur = end;
|
|
break;
|
|
case 'T' :
|
|
*mode = GRN_OP_TERM_EXTRACT;
|
|
start = ++end;
|
|
*option = grn_atoi(start, q->str_end, (const char **)&end);
|
|
if (start == end) { *option = DEFAULT_TERM_EXTRACT_POLICY; }
|
|
q->cur = end;
|
|
break;
|
|
case 'X' : /* force exact mode */
|
|
op->op = GRN_OP_AND;
|
|
*mode = GRN_OP_EXACT;
|
|
*option = 0;
|
|
start = ++end;
|
|
q->cur = end;
|
|
break;
|
|
default :
|
|
found = GRN_FALSE;
|
|
break;
|
|
}
|
|
return found;
|
|
}
|
|
|
|
static grn_rc
|
|
get_token(grn_ctx *ctx, efs_info *q, efs_op *op, grn_obj *column, grn_operator mode)
|
|
{
|
|
int option = 0;
|
|
op->op = q->default_op;
|
|
op->weight = DEFAULT_WEIGHT;
|
|
for (;;) {
|
|
skip_space(ctx, q);
|
|
if (q->cur >= q->str_end) { return GRN_END_OF_DATA; }
|
|
switch (*q->cur) {
|
|
case '\0' :
|
|
return GRN_END_OF_DATA;
|
|
break;
|
|
case GRN_QUERY_PARENR :
|
|
q->cur++;
|
|
return GRN_END_OF_DATA;
|
|
break;
|
|
case GRN_QUERY_QUOTEL :
|
|
q->cur++;
|
|
return get_phrase(ctx, q, column, mode, option);
|
|
break;
|
|
case GRN_QUERY_PREFIX :
|
|
q->cur++;
|
|
get_op(q, op, &mode, &option);
|
|
break;
|
|
case GRN_QUERY_AND :
|
|
q->cur++;
|
|
op->op = GRN_OP_AND;
|
|
break;
|
|
case GRN_QUERY_AND_NOT :
|
|
q->cur++;
|
|
op->op = GRN_OP_AND_NOT;
|
|
break;
|
|
case GRN_QUERY_ADJ_INC :
|
|
q->cur++;
|
|
if (op->weight < 127) { op->weight++; }
|
|
op->op = GRN_OP_ADJUST;
|
|
break;
|
|
case GRN_QUERY_ADJ_DEC :
|
|
q->cur++;
|
|
if (op->weight > -128) { op->weight--; }
|
|
op->op = GRN_OP_ADJUST;
|
|
break;
|
|
case GRN_QUERY_ADJ_NEG :
|
|
q->cur++;
|
|
op->op = GRN_OP_ADJUST;
|
|
op->weight = -1;
|
|
break;
|
|
case GRN_QUERY_PARENL :
|
|
q->cur++;
|
|
return get_expr(ctx, q, column, mode);
|
|
break;
|
|
case 'O' :
|
|
if (q->cur[1] == 'R' && q->cur[2] == ' ') {
|
|
q->cur += 2;
|
|
op->op = GRN_OP_OR;
|
|
break;
|
|
}
|
|
/* fallthru */
|
|
default :
|
|
return get_word(ctx, q, column, mode, option);
|
|
break;
|
|
}
|
|
}
|
|
return GRN_SUCCESS;
|
|
}
|
|
|
|
static grn_rc
|
|
get_expr(grn_ctx *ctx, efs_info *q, grn_obj *column, grn_operator mode)
|
|
{
|
|
efs_op op;
|
|
grn_rc rc = get_token(ctx, q, &op, column, mode);
|
|
if (rc) { return rc; }
|
|
while (!(rc = get_token(ctx, q, &op, column, mode))) {
|
|
if (op.op == GRN_OP_ADJUST) {
|
|
grn_expr_append_const_int(ctx, q->e, op.weight, GRN_OP_PUSH, 1);
|
|
grn_expr_append_op(ctx, q->e, op.op, 3);
|
|
} else {
|
|
grn_expr_append_op(ctx, q->e, op.op, 2);
|
|
}
|
|
}
|
|
return rc;
|
|
}
|
|
|
|
#define DISABLE_UNUSED_CODE 1
|
|
#ifndef DISABLE_UNUSED_CODE
|
|
static const char *
|
|
get_weight_vector(grn_ctx *ctx, efs_info *query, const char *source)
|
|
{
|
|
const char *p;
|
|
|
|
if (!query->opt.weight_vector &&
|
|
!query->weight_set &&
|
|
!(query->opt.weight_vector = GRN_CALLOC(sizeof(int) * DEFAULT_WEIGHT_VECTOR_SIZE))) {
|
|
GRN_LOG(ctx, GRN_LOG_ALERT, "get_weight_vector malloc fail");
|
|
return source;
|
|
}
|
|
for (p = source; p < query->str_end; ) {
|
|
unsigned int key;
|
|
int value;
|
|
|
|
/* key, key is not zero */
|
|
key = grn_atoui(p, query->str_end, &p);
|
|
if (!key || key > GRN_ID_MAX) { break; }
|
|
|
|
/* value */
|
|
if (*p == ':') {
|
|
p++;
|
|
value = grn_atoi(p, query->str_end, &p);
|
|
} else {
|
|
value = 1;
|
|
}
|
|
|
|
if (query->weight_set) {
|
|
int *pval;
|
|
if (grn_hash_add(ctx, query->weight_set, &key, sizeof(unsigned int), (void **)&pval, NULL)) {
|
|
*pval = value;
|
|
}
|
|
} else if (key < DEFAULT_WEIGHT_VECTOR_SIZE) {
|
|
query->opt.weight_vector[key - 1] = value;
|
|
} else {
|
|
GRN_FREE(query->opt.weight_vector);
|
|
query->opt.weight_vector = NULL;
|
|
if (!(query->weight_set = grn_hash_create(ctx, NULL, sizeof(unsigned int), sizeof(int),
|
|
0))) {
|
|
return source;
|
|
}
|
|
p = source; /* reparse */
|
|
continue;
|
|
}
|
|
if (*p != ',') { break; }
|
|
p++;
|
|
}
|
|
return p;
|
|
}
|
|
|
|
static void
|
|
get_pragma(grn_ctx *ctx, efs_info *q)
|
|
{
|
|
const char *start, *end = q->cur;
|
|
while (end < q->str_end && *end == GRN_QUERY_PREFIX) {
|
|
if (++end >= q->str_end) { break; }
|
|
switch (*end) {
|
|
case 'E' :
|
|
start = ++end;
|
|
q->escalation_threshold = grn_atoi(start, q->str_end, (const char **)&end);
|
|
while (end < q->str_end && (('0' <= *end && *end <= '9') || *end == '-')) { end++; }
|
|
if (*end == ',') {
|
|
start = ++end;
|
|
q->escalation_decaystep = grn_atoi(start, q->str_end, (const char **)&end);
|
|
}
|
|
q->cur = end;
|
|
break;
|
|
case 'D' :
|
|
start = ++end;
|
|
while (end < q->str_end && *end != GRN_QUERY_PREFIX && !grn_isspace(end, ctx->encoding)) {
|
|
end++;
|
|
}
|
|
if (end > start) {
|
|
switch (*start) {
|
|
case 'O' :
|
|
q->default_op = GRN_OP_OR;
|
|
break;
|
|
case GRN_QUERY_AND :
|
|
q->default_op = GRN_OP_AND;
|
|
break;
|
|
case GRN_QUERY_AND_NOT :
|
|
q->default_op = GRN_OP_AND_NOT;
|
|
break;
|
|
case GRN_QUERY_ADJ_INC :
|
|
q->default_op = GRN_OP_ADJUST;
|
|
break;
|
|
}
|
|
}
|
|
q->cur = end;
|
|
break;
|
|
case 'W' :
|
|
start = ++end;
|
|
end = (char *)get_weight_vector(ctx, q, start);
|
|
q->cur = end;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
static int
|
|
section_weight_cb(grn_ctx *ctx, grn_hash *r, const void *rid, int sid, void *arg)
|
|
{
|
|
int *w;
|
|
grn_hash *s = (grn_hash *)arg;
|
|
if (s && grn_hash_get(ctx, s, &sid, sizeof(grn_id), (void **)&w)) {
|
|
return *w;
|
|
} else {
|
|
return 0;
|
|
}
|
|
}
|
|
#endif
|
|
|
|
#include "grn_ecmascript.h"
|
|
#include "grn_ecmascript.c"
|
|
|
|
static grn_rc
|
|
grn_expr_parser_open(grn_ctx *ctx)
|
|
{
|
|
if (!ctx->impl->parser) {
|
|
yyParser *pParser = GRN_MALLOCN(yyParser, 1);
|
|
if (pParser) {
|
|
pParser->yyidx = -1;
|
|
#if YYSTACKDEPTH<=0
|
|
yyGrowStack(pParser);
|
|
#endif
|
|
ctx->impl->parser = pParser;
|
|
}
|
|
}
|
|
return ctx->rc;
|
|
}
|
|
|
|
#define PARSE(token) grn_expr_parser(ctx->impl->parser, (token), 0, q)
|
|
|
|
static void
|
|
accept_query_string(grn_ctx *ctx, efs_info *efsi,
|
|
const char *str, unsigned int str_size)
|
|
{
|
|
grn_obj *column, *token;
|
|
grn_operator mode;
|
|
|
|
GRN_PTR_PUT(ctx, &efsi->token_stack,
|
|
grn_expr_add_str(ctx, efsi->e, str, str_size));
|
|
{
|
|
efs_info *q = efsi;
|
|
PARSE(GRN_EXPR_TOKEN_QSTRING);
|
|
}
|
|
|
|
GRN_PTR_POP(&efsi->token_stack, token);
|
|
column = grn_ptr_value_at(&efsi->column_stack, -1);
|
|
grn_expr_append_const(efsi->ctx, efsi->e, column, GRN_OP_GET_VALUE, 1);
|
|
grn_expr_append_obj(efsi->ctx, efsi->e, token, GRN_OP_PUSH, 1);
|
|
|
|
mode = grn_int32_value_at(&efsi->mode_stack, -1);
|
|
switch (mode) {
|
|
case GRN_OP_NEAR :
|
|
case GRN_OP_NEAR2 :
|
|
{
|
|
int max_interval;
|
|
max_interval = grn_int32_value_at(&efsi->max_interval_stack, -1);
|
|
grn_expr_append_const_int(efsi->ctx, efsi->e, max_interval,
|
|
GRN_OP_PUSH, 1);
|
|
grn_expr_append_op(efsi->ctx, efsi->e, mode, 3);
|
|
}
|
|
break;
|
|
case GRN_OP_SIMILAR :
|
|
{
|
|
int similarity_threshold;
|
|
similarity_threshold =
|
|
grn_int32_value_at(&efsi->similarity_threshold_stack, -1);
|
|
grn_expr_append_const_int(efsi->ctx, efsi->e, similarity_threshold,
|
|
GRN_OP_PUSH, 1);
|
|
grn_expr_append_op(efsi->ctx, efsi->e, mode, 3);
|
|
}
|
|
break;
|
|
default :
|
|
grn_expr_append_op(efsi->ctx, efsi->e, mode, 2);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static grn_rc
|
|
get_word_(grn_ctx *ctx, efs_info *q)
|
|
{
|
|
const char *end;
|
|
unsigned int len;
|
|
GRN_BULK_REWIND(&q->buf);
|
|
for (end = q->cur;; ) {
|
|
/* null check and length check */
|
|
if (!(len = grn_charlen(ctx, end, q->str_end))) {
|
|
q->cur = q->str_end;
|
|
break;
|
|
}
|
|
if (grn_isspace(end, ctx->encoding) ||
|
|
*end == GRN_QUERY_PARENL || *end == GRN_QUERY_PARENR) {
|
|
q->cur = end;
|
|
break;
|
|
}
|
|
if (q->flags & GRN_EXPR_ALLOW_COLUMN && *end == GRN_QUERY_COLUMN) {
|
|
grn_operator mode;
|
|
grn_obj *c = grn_obj_column(ctx, q->table,
|
|
GRN_TEXT_VALUE(&q->buf),
|
|
GRN_TEXT_LEN(&q->buf));
|
|
if (c && end + 1 < q->str_end) {
|
|
// efs_op op;
|
|
switch (end[1]) {
|
|
case '!' :
|
|
mode = GRN_OP_NOT_EQUAL;
|
|
q->cur = end + 2;
|
|
break;
|
|
case '=' :
|
|
if (q->flags & GRN_EXPR_ALLOW_UPDATE) {
|
|
mode = GRN_OP_ASSIGN;
|
|
q->cur = end + 2;
|
|
} else {
|
|
mode = GRN_OP_EQUAL;
|
|
q->cur = end + 1;
|
|
}
|
|
break;
|
|
case '<' :
|
|
if (end + 2 < q->str_end && end[2] == '=') {
|
|
mode = GRN_OP_LESS_EQUAL;
|
|
q->cur = end + 3;
|
|
} else {
|
|
mode = GRN_OP_LESS;
|
|
q->cur = end + 2;
|
|
}
|
|
break;
|
|
case '>' :
|
|
if (end + 2 < q->str_end && end[2] == '=') {
|
|
mode = GRN_OP_GREATER_EQUAL;
|
|
q->cur = end + 3;
|
|
} else {
|
|
mode = GRN_OP_GREATER;
|
|
q->cur = end + 2;
|
|
}
|
|
break;
|
|
case '@' :
|
|
mode = GRN_OP_MATCH;
|
|
q->cur = end + 2;
|
|
break;
|
|
case '^' :
|
|
mode = GRN_OP_PREFIX;
|
|
q->cur = end + 2;
|
|
break;
|
|
case '$' :
|
|
mode = GRN_OP_SUFFIX;
|
|
q->cur = end + 2;
|
|
break;
|
|
default :
|
|
mode = GRN_OP_EQUAL;
|
|
q->cur = end + 1;
|
|
break;
|
|
}
|
|
} else {
|
|
ERR(GRN_INVALID_ARGUMENT, "column lookup failed");
|
|
q->cur = q->str_end;
|
|
return ctx->rc;
|
|
}
|
|
PARSE(GRN_EXPR_TOKEN_IDENTIFIER);
|
|
PARSE(GRN_EXPR_TOKEN_RELATIVE_OP);
|
|
|
|
grn_expr_take_obj(ctx, q->e, c);
|
|
GRN_PTR_PUT(ctx, &q->column_stack, c);
|
|
GRN_INT32_PUT(ctx, &q->mode_stack, mode);
|
|
|
|
return GRN_SUCCESS;
|
|
} else if (GRN_TEXT_LEN(&q->buf) > 0 && *end == GRN_QUERY_PREFIX) {
|
|
q->cur = end + 1;
|
|
GRN_INT32_PUT(ctx, &q->mode_stack, GRN_OP_PREFIX);
|
|
break;
|
|
} else if (*end == GRN_QUERY_ESCAPE) {
|
|
end += len;
|
|
if (!(len = grn_charlen(ctx, end, q->str_end))) {
|
|
q->cur = q->str_end;
|
|
break;
|
|
}
|
|
}
|
|
GRN_TEXT_PUT(ctx, &q->buf, end, len);
|
|
end += len;
|
|
}
|
|
accept_query_string(ctx, q, GRN_TEXT_VALUE(&q->buf), GRN_TEXT_LEN(&q->buf));
|
|
|
|
return GRN_SUCCESS;
|
|
}
|
|
|
|
static grn_rc
|
|
parse_query(grn_ctx *ctx, efs_info *q)
|
|
{
|
|
int option = 0;
|
|
grn_operator mode;
|
|
efs_op op_, *op = &op_;
|
|
grn_bool first_token = GRN_TRUE;
|
|
grn_bool block_started = GRN_FALSE;
|
|
|
|
op->op = q->default_op;
|
|
op->weight = DEFAULT_WEIGHT;
|
|
while (!ctx->rc) {
|
|
skip_space(ctx, q);
|
|
if (q->cur >= q->str_end) { goto exit; }
|
|
switch (*q->cur) {
|
|
case '\0' :
|
|
goto exit;
|
|
break;
|
|
case GRN_QUERY_PARENR :
|
|
PARSE(GRN_EXPR_TOKEN_PARENR);
|
|
q->cur++;
|
|
break;
|
|
case GRN_QUERY_QUOTEL :
|
|
q->cur++;
|
|
|
|
{
|
|
const char *start, *s;
|
|
start = s = q->cur;
|
|
GRN_BULK_REWIND(&q->buf);
|
|
while (1) {
|
|
unsigned int len;
|
|
if (s >= q->str_end) {
|
|
q->cur = s;
|
|
break;
|
|
}
|
|
len = grn_charlen(ctx, s, q->str_end);
|
|
if (len == 0) {
|
|
/* invalid string containing malformed multibyte char */
|
|
goto exit;
|
|
} else if (len == 1) {
|
|
if (*s == GRN_QUERY_QUOTER) {
|
|
q->cur = s + 1;
|
|
break;
|
|
} else if (*s == GRN_QUERY_ESCAPE && s + 1 < q->str_end) {
|
|
s++;
|
|
len = grn_charlen(ctx, s, q->str_end);
|
|
}
|
|
}
|
|
GRN_TEXT_PUT(ctx, &q->buf, s, len);
|
|
s += len;
|
|
|
|
}
|
|
accept_query_string(ctx, q,
|
|
GRN_TEXT_VALUE(&q->buf), GRN_TEXT_LEN(&q->buf));
|
|
}
|
|
|
|
break;
|
|
case GRN_QUERY_PREFIX :
|
|
q->cur++;
|
|
if (get_op(q, op, &mode, &option)) {
|
|
switch (mode) {
|
|
case GRN_OP_NEAR :
|
|
case GRN_OP_NEAR2 :
|
|
GRN_INT32_PUT(ctx, &q->max_interval_stack, option);
|
|
break;
|
|
case GRN_OP_SIMILAR :
|
|
GRN_INT32_PUT(ctx, &q->similarity_threshold_stack, option);
|
|
break;
|
|
default :
|
|
break;
|
|
}
|
|
GRN_INT32_PUT(ctx, &q->mode_stack, mode);
|
|
PARSE(GRN_EXPR_TOKEN_RELATIVE_OP);
|
|
} else {
|
|
q->cur--;
|
|
get_word_(ctx, q);
|
|
}
|
|
break;
|
|
case GRN_QUERY_AND :
|
|
if (!first_token) {
|
|
op->op = GRN_OP_AND;
|
|
PARSE(GRN_EXPR_TOKEN_LOGICAL_AND);
|
|
}
|
|
q->cur++;
|
|
break;
|
|
case GRN_QUERY_AND_NOT :
|
|
if (first_token && (q->flags & GRN_EXPR_ALLOW_LEADING_NOT)) {
|
|
grn_obj *all_records = grn_ctx_get(ctx, "all_records", 11);
|
|
if (all_records) {
|
|
/* dummy token */
|
|
PARSE(GRN_EXPR_TOKEN_QSTRING);
|
|
grn_expr_append_obj(ctx, q->e, all_records, GRN_OP_PUSH, 1);
|
|
grn_expr_append_op(ctx, q->e, GRN_OP_CALL, 0);
|
|
}
|
|
}
|
|
op->op = GRN_OP_AND_NOT;
|
|
PARSE(GRN_EXPR_TOKEN_LOGICAL_AND_NOT);
|
|
q->cur++;
|
|
break;
|
|
case GRN_QUERY_ADJ_INC :
|
|
if (op->weight < 127) { op->weight++; }
|
|
op->op = GRN_OP_ADJUST;
|
|
PARSE(GRN_EXPR_TOKEN_ADJUST);
|
|
q->cur++;
|
|
break;
|
|
case GRN_QUERY_ADJ_DEC :
|
|
if (op->weight > -128) { op->weight--; }
|
|
op->op = GRN_OP_ADJUST;
|
|
PARSE(GRN_EXPR_TOKEN_ADJUST);
|
|
q->cur++;
|
|
break;
|
|
case GRN_QUERY_ADJ_NEG :
|
|
op->op = GRN_OP_ADJUST;
|
|
op->weight = -1;
|
|
PARSE(GRN_EXPR_TOKEN_ADJUST);
|
|
q->cur++;
|
|
break;
|
|
case GRN_QUERY_PARENL :
|
|
PARSE(GRN_EXPR_TOKEN_PARENL);
|
|
q->cur++;
|
|
block_started = GRN_TRUE;
|
|
break;
|
|
case 'O' :
|
|
if (q->cur[1] == 'R' && q->cur[2] == ' ') {
|
|
PARSE(GRN_EXPR_TOKEN_LOGICAL_OR);
|
|
q->cur += 2;
|
|
break;
|
|
}
|
|
/* fallthru */
|
|
default :
|
|
get_word_(ctx, q);
|
|
break;
|
|
}
|
|
first_token = block_started;
|
|
block_started = GRN_FALSE;
|
|
}
|
|
exit :
|
|
PARSE(0);
|
|
return GRN_SUCCESS;
|
|
}
|
|
|
|
static grn_rc
|
|
get_string(grn_ctx *ctx, efs_info *q, char quote)
|
|
{
|
|
const char *s;
|
|
unsigned int len;
|
|
grn_rc rc = GRN_END_OF_DATA;
|
|
GRN_BULK_REWIND(&q->buf);
|
|
for (s = q->cur + 1; s < q->str_end; s += len) {
|
|
if (!(len = grn_charlen(ctx, s, q->str_end))) { break; }
|
|
if (len == 1) {
|
|
if (*s == quote) {
|
|
s++;
|
|
rc = GRN_SUCCESS;
|
|
break;
|
|
}
|
|
if (*s == GRN_QUERY_ESCAPE && s + 1 < q->str_end) {
|
|
s++;
|
|
if (!(len = grn_charlen(ctx, s, q->str_end))) { break; }
|
|
}
|
|
}
|
|
GRN_TEXT_PUT(ctx, &q->buf, s, len);
|
|
}
|
|
q->cur = s;
|
|
return rc;
|
|
}
|
|
|
|
static grn_obj *
|
|
resolve_top_level_name(grn_ctx *ctx, const char *name, unsigned int name_size)
|
|
{
|
|
unsigned int i;
|
|
unsigned int first_delimiter_position = 0;
|
|
unsigned int n_delimiters = 0;
|
|
grn_obj *top_level_object;
|
|
grn_obj *object;
|
|
|
|
for (i = 0; i < name_size; i++) {
|
|
if (name[i] != GRN_DB_DELIMITER) {
|
|
continue;
|
|
}
|
|
|
|
if (n_delimiters == 0) {
|
|
first_delimiter_position = i;
|
|
}
|
|
n_delimiters++;
|
|
}
|
|
|
|
if (n_delimiters < 2) {
|
|
return grn_ctx_get(ctx, name, name_size);
|
|
}
|
|
|
|
top_level_object = grn_ctx_get(ctx, name, first_delimiter_position);
|
|
if (!top_level_object) {
|
|
return NULL;
|
|
}
|
|
object = grn_obj_column(ctx, top_level_object,
|
|
name + first_delimiter_position + 1,
|
|
name_size - first_delimiter_position - 1);
|
|
grn_obj_unlink(ctx, top_level_object);
|
|
return object;
|
|
}
|
|
|
|
static grn_rc
|
|
get_identifier(grn_ctx *ctx, efs_info *q)
|
|
{
|
|
const char *s;
|
|
unsigned int len;
|
|
grn_rc rc = GRN_SUCCESS;
|
|
for (s = q->cur; s < q->str_end; s += len) {
|
|
if (!(len = grn_charlen(ctx, s, q->str_end))) {
|
|
rc = GRN_END_OF_DATA;
|
|
goto exit;
|
|
}
|
|
if (grn_isspace(s, ctx->encoding)) { goto done; }
|
|
if (len == 1) {
|
|
switch (*s) {
|
|
case '\0' : case '(' : case ')' : case '{' : case '}' :
|
|
case '[' : case ']' : case ',' : case ':' : case '@' :
|
|
case '?' : case '"' : case '*' : case '+' : case '-' :
|
|
case '|' : case '/' : case '%' : case '!' : case '^' :
|
|
case '&' : case '>' : case '<' : case '=' : case '~' :
|
|
/* case '.' : */
|
|
goto done;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
done :
|
|
len = s - q->cur;
|
|
switch (*q->cur) {
|
|
case 'd' :
|
|
if (len == 6 && !memcmp(q->cur, "delete", 6)) {
|
|
PARSE(GRN_EXPR_TOKEN_DELETE);
|
|
goto exit;
|
|
}
|
|
break;
|
|
case 'f' :
|
|
if (len == 5 && !memcmp(q->cur, "false", 5)) {
|
|
grn_obj buf;
|
|
PARSE(GRN_EXPR_TOKEN_BOOLEAN);
|
|
GRN_BOOL_INIT(&buf, 0);
|
|
GRN_BOOL_SET(ctx, &buf, 0);
|
|
grn_expr_append_const(ctx, q->e, &buf, GRN_OP_PUSH, 1);
|
|
GRN_OBJ_FIN(ctx, &buf);
|
|
goto exit;
|
|
}
|
|
break;
|
|
case 'i' :
|
|
if (len == 2 && !memcmp(q->cur, "in", 2)) {
|
|
PARSE(GRN_EXPR_TOKEN_IN);
|
|
goto exit;
|
|
}
|
|
break;
|
|
case 'n' :
|
|
if (len == 4 && !memcmp(q->cur, "null", 4)) {
|
|
grn_obj buf;
|
|
PARSE(GRN_EXPR_TOKEN_NULL);
|
|
GRN_VOID_INIT(&buf);
|
|
grn_expr_append_const(ctx, q->e, &buf, GRN_OP_PUSH, 1);
|
|
GRN_OBJ_FIN(ctx, &buf);
|
|
goto exit;
|
|
}
|
|
break;
|
|
case 't' :
|
|
if (len == 4 && !memcmp(q->cur, "true", 4)) {
|
|
grn_obj buf;
|
|
PARSE(GRN_EXPR_TOKEN_BOOLEAN);
|
|
GRN_BOOL_INIT(&buf, 0);
|
|
GRN_BOOL_SET(ctx, &buf, 1);
|
|
grn_expr_append_const(ctx, q->e, &buf, GRN_OP_PUSH, 1);
|
|
GRN_OBJ_FIN(ctx, &buf);
|
|
goto exit;
|
|
}
|
|
break;
|
|
}
|
|
{
|
|
grn_obj *obj;
|
|
const char *name = q->cur;
|
|
unsigned int name_size = s - q->cur;
|
|
if ((obj = grn_expr_get_var(ctx, q->e, name, name_size))) {
|
|
PARSE(GRN_EXPR_TOKEN_IDENTIFIER);
|
|
grn_expr_append_obj(ctx, q->e, obj, GRN_OP_PUSH, 1);
|
|
goto exit;
|
|
}
|
|
if ((obj = grn_obj_column(ctx, q->table, name, name_size))) {
|
|
grn_expr_take_obj(ctx, q->e, obj);
|
|
PARSE(GRN_EXPR_TOKEN_IDENTIFIER);
|
|
grn_expr_append_obj(ctx, q->e, obj, GRN_OP_GET_VALUE, 1);
|
|
goto exit;
|
|
}
|
|
if ((obj = resolve_top_level_name(ctx, name, name_size))) {
|
|
grn_expr_take_obj(ctx, q->e, obj);
|
|
PARSE(GRN_EXPR_TOKEN_IDENTIFIER);
|
|
grn_expr_append_obj(ctx, q->e, obj, GRN_OP_PUSH, 1);
|
|
goto exit;
|
|
}
|
|
if (q->flags & GRN_EXPR_SYNTAX_OUTPUT_COLUMNS) {
|
|
PARSE(GRN_EXPR_TOKEN_NONEXISTENT_COLUMN);
|
|
} else {
|
|
rc = GRN_SYNTAX_ERROR;
|
|
}
|
|
}
|
|
exit :
|
|
q->cur = s;
|
|
return rc;
|
|
}
|
|
|
|
static void
|
|
set_tos_minor_to_curr(grn_ctx *ctx, efs_info *q)
|
|
{
|
|
yyParser *pParser = ctx->impl->parser;
|
|
yyStackEntry *yytos = &pParser->yystack[pParser->yyidx];
|
|
yytos->minor.yy0 = ((grn_expr *)(q->e))->codes_curr;
|
|
}
|
|
|
|
static grn_rc
|
|
parse_script(grn_ctx *ctx, efs_info *q)
|
|
{
|
|
grn_rc rc = GRN_SUCCESS;
|
|
for (;;) {
|
|
skip_space(ctx, q);
|
|
if (q->cur >= q->str_end) { rc = GRN_END_OF_DATA; goto exit; }
|
|
switch (*q->cur) {
|
|
case '\0' :
|
|
rc = GRN_END_OF_DATA;
|
|
goto exit;
|
|
break;
|
|
case '(' :
|
|
PARSE(GRN_EXPR_TOKEN_PARENL);
|
|
q->cur++;
|
|
break;
|
|
case ')' :
|
|
PARSE(GRN_EXPR_TOKEN_PARENR);
|
|
q->cur++;
|
|
break;
|
|
case '{' :
|
|
PARSE(GRN_EXPR_TOKEN_BRACEL);
|
|
q->cur++;
|
|
break;
|
|
case '}' :
|
|
PARSE(GRN_EXPR_TOKEN_BRACER);
|
|
q->cur++;
|
|
break;
|
|
case '[' :
|
|
PARSE(GRN_EXPR_TOKEN_BRACKETL);
|
|
q->cur++;
|
|
break;
|
|
case ']' :
|
|
PARSE(GRN_EXPR_TOKEN_BRACKETR);
|
|
q->cur++;
|
|
break;
|
|
case ',' :
|
|
PARSE(GRN_EXPR_TOKEN_COMMA);
|
|
q->cur++;
|
|
break;
|
|
case '.' :
|
|
PARSE(GRN_EXPR_TOKEN_DOT);
|
|
q->cur++;
|
|
break;
|
|
case ':' :
|
|
PARSE(GRN_EXPR_TOKEN_COLON);
|
|
q->cur++;
|
|
set_tos_minor_to_curr(ctx, q);
|
|
grn_expr_append_op(ctx, q->e, GRN_OP_JUMP, 0);
|
|
break;
|
|
case '@' :
|
|
switch (q->cur[1]) {
|
|
case '^' :
|
|
PARSE(GRN_EXPR_TOKEN_PREFIX);
|
|
q->cur += 2;
|
|
break;
|
|
case '$' :
|
|
PARSE(GRN_EXPR_TOKEN_SUFFIX);
|
|
q->cur += 2;
|
|
break;
|
|
default :
|
|
PARSE(GRN_EXPR_TOKEN_MATCH);
|
|
q->cur++;
|
|
break;
|
|
}
|
|
break;
|
|
case '~' :
|
|
PARSE(GRN_EXPR_TOKEN_BITWISE_NOT);
|
|
q->cur++;
|
|
break;
|
|
case '?' :
|
|
PARSE(GRN_EXPR_TOKEN_QUESTION);
|
|
q->cur++;
|
|
set_tos_minor_to_curr(ctx, q);
|
|
grn_expr_append_op(ctx, q->e, GRN_OP_CJUMP, 0);
|
|
break;
|
|
case '"' :
|
|
if ((rc = get_string(ctx, q, '"'))) { goto exit; }
|
|
PARSE(GRN_EXPR_TOKEN_STRING);
|
|
grn_expr_append_const(ctx, q->e, &q->buf, GRN_OP_PUSH, 1);
|
|
break;
|
|
case '\'' :
|
|
if ((rc = get_string(ctx, q, '\''))) { goto exit; }
|
|
PARSE(GRN_EXPR_TOKEN_STRING);
|
|
grn_expr_append_const(ctx, q->e, &q->buf, GRN_OP_PUSH, 1);
|
|
break;
|
|
case '*' :
|
|
switch (q->cur[1]) {
|
|
case 'N' :
|
|
PARSE(GRN_EXPR_TOKEN_NEAR);
|
|
q->cur += 2;
|
|
break;
|
|
case 'S' :
|
|
PARSE(GRN_EXPR_TOKEN_SIMILAR);
|
|
q->cur += 2;
|
|
break;
|
|
case 'T' :
|
|
PARSE(GRN_EXPR_TOKEN_TERM_EXTRACT);
|
|
q->cur += 2;
|
|
break;
|
|
case '>' :
|
|
PARSE(GRN_EXPR_TOKEN_ADJUST);
|
|
q->cur += 2;
|
|
break;
|
|
case '<' :
|
|
PARSE(GRN_EXPR_TOKEN_ADJUST);
|
|
q->cur += 2;
|
|
break;
|
|
case '~' :
|
|
PARSE(GRN_EXPR_TOKEN_ADJUST);
|
|
q->cur += 2;
|
|
break;
|
|
case '=' :
|
|
if (q->flags & GRN_EXPR_ALLOW_UPDATE) {
|
|
PARSE(GRN_EXPR_TOKEN_STAR_ASSIGN);
|
|
q->cur += 2;
|
|
} else {
|
|
ERR(GRN_UPDATE_NOT_ALLOWED,
|
|
"'*=' is not allowed (%.*s)", (int)(q->str_end - q->str), q->str);
|
|
}
|
|
break;
|
|
default :
|
|
PARSE(GRN_EXPR_TOKEN_STAR);
|
|
q->cur++;
|
|
break;
|
|
}
|
|
break;
|
|
case '+' :
|
|
switch (q->cur[1]) {
|
|
case '+' :
|
|
if (q->flags & GRN_EXPR_ALLOW_UPDATE) {
|
|
PARSE(GRN_EXPR_TOKEN_INCR);
|
|
q->cur += 2;
|
|
} else {
|
|
ERR(GRN_UPDATE_NOT_ALLOWED,
|
|
"'++' is not allowed (%.*s)", (int)(q->str_end - q->str), q->str);
|
|
}
|
|
break;
|
|
case '=' :
|
|
if (q->flags & GRN_EXPR_ALLOW_UPDATE) {
|
|
PARSE(GRN_EXPR_TOKEN_PLUS_ASSIGN);
|
|
q->cur += 2;
|
|
} else {
|
|
ERR(GRN_UPDATE_NOT_ALLOWED,
|
|
"'+=' is not allowed (%.*s)", (int)(q->str_end - q->str), q->str);
|
|
}
|
|
break;
|
|
default :
|
|
PARSE(GRN_EXPR_TOKEN_PLUS);
|
|
q->cur++;
|
|
break;
|
|
}
|
|
break;
|
|
case '-' :
|
|
switch (q->cur[1]) {
|
|
case '-' :
|
|
if (q->flags & GRN_EXPR_ALLOW_UPDATE) {
|
|
PARSE(GRN_EXPR_TOKEN_DECR);
|
|
q->cur += 2;
|
|
} else {
|
|
ERR(GRN_UPDATE_NOT_ALLOWED,
|
|
"'--' is not allowed (%.*s)", (int)(q->str_end - q->str), q->str);
|
|
}
|
|
break;
|
|
case '=' :
|
|
if (q->flags & GRN_EXPR_ALLOW_UPDATE) {
|
|
PARSE(GRN_EXPR_TOKEN_MINUS_ASSIGN);
|
|
q->cur += 2;
|
|
} else {
|
|
ERR(GRN_UPDATE_NOT_ALLOWED,
|
|
"'-=' is not allowed (%.*s)", (int)(q->str_end - q->str), q->str);
|
|
}
|
|
break;
|
|
default :
|
|
PARSE(GRN_EXPR_TOKEN_MINUS);
|
|
q->cur++;
|
|
break;
|
|
}
|
|
break;
|
|
case '|' :
|
|
switch (q->cur[1]) {
|
|
case '|' :
|
|
PARSE(GRN_EXPR_TOKEN_LOGICAL_OR);
|
|
q->cur += 2;
|
|
break;
|
|
case '=' :
|
|
if (q->flags & GRN_EXPR_ALLOW_UPDATE) {
|
|
PARSE(GRN_EXPR_TOKEN_OR_ASSIGN);
|
|
q->cur += 2;
|
|
} else {
|
|
ERR(GRN_UPDATE_NOT_ALLOWED,
|
|
"'|=' is not allowed (%.*s)", (int)(q->str_end - q->str), q->str);
|
|
}
|
|
break;
|
|
default :
|
|
PARSE(GRN_EXPR_TOKEN_BITWISE_OR);
|
|
q->cur++;
|
|
break;
|
|
}
|
|
break;
|
|
case '/' :
|
|
switch (q->cur[1]) {
|
|
case '=' :
|
|
if (q->flags & GRN_EXPR_ALLOW_UPDATE) {
|
|
PARSE(GRN_EXPR_TOKEN_SLASH_ASSIGN);
|
|
q->cur += 2;
|
|
} else {
|
|
ERR(GRN_UPDATE_NOT_ALLOWED,
|
|
"'/=' is not allowed (%.*s)", (int)(q->str_end - q->str), q->str);
|
|
}
|
|
break;
|
|
default :
|
|
PARSE(GRN_EXPR_TOKEN_SLASH);
|
|
q->cur++;
|
|
break;
|
|
}
|
|
break;
|
|
case '%' :
|
|
switch (q->cur[1]) {
|
|
case '=' :
|
|
if (q->flags & GRN_EXPR_ALLOW_UPDATE) {
|
|
PARSE(GRN_EXPR_TOKEN_MOD_ASSIGN);
|
|
q->cur += 2;
|
|
} else {
|
|
ERR(GRN_UPDATE_NOT_ALLOWED,
|
|
"'%%=' is not allowed (%.*s)", (int)(q->str_end - q->str), q->str);
|
|
}
|
|
break;
|
|
default :
|
|
PARSE(GRN_EXPR_TOKEN_MOD);
|
|
q->cur++;
|
|
break;
|
|
}
|
|
break;
|
|
case '!' :
|
|
switch (q->cur[1]) {
|
|
case '=' :
|
|
PARSE(GRN_EXPR_TOKEN_NOT_EQUAL);
|
|
q->cur += 2;
|
|
break;
|
|
default :
|
|
PARSE(GRN_EXPR_TOKEN_NOT);
|
|
q->cur++;
|
|
break;
|
|
}
|
|
break;
|
|
case '^' :
|
|
switch (q->cur[1]) {
|
|
case '=' :
|
|
if (q->flags & GRN_EXPR_ALLOW_UPDATE) {
|
|
q->cur += 2;
|
|
PARSE(GRN_EXPR_TOKEN_XOR_ASSIGN);
|
|
} else {
|
|
ERR(GRN_UPDATE_NOT_ALLOWED,
|
|
"'^=' is not allowed (%.*s)", (int)(q->str_end - q->str), q->str);
|
|
}
|
|
break;
|
|
default :
|
|
PARSE(GRN_EXPR_TOKEN_BITWISE_XOR);
|
|
q->cur++;
|
|
break;
|
|
}
|
|
break;
|
|
case '&' :
|
|
switch (q->cur[1]) {
|
|
case '&' :
|
|
PARSE(GRN_EXPR_TOKEN_LOGICAL_AND);
|
|
q->cur += 2;
|
|
break;
|
|
case '=' :
|
|
if (q->flags & GRN_EXPR_ALLOW_UPDATE) {
|
|
PARSE(GRN_EXPR_TOKEN_AND_ASSIGN);
|
|
q->cur += 2;
|
|
} else {
|
|
ERR(GRN_UPDATE_NOT_ALLOWED,
|
|
"'&=' is not allowed (%.*s)", (int)(q->str_end - q->str), q->str);
|
|
}
|
|
break;
|
|
case '!' :
|
|
PARSE(GRN_EXPR_TOKEN_LOGICAL_AND_NOT);
|
|
q->cur += 2;
|
|
break;
|
|
default :
|
|
PARSE(GRN_EXPR_TOKEN_BITWISE_AND);
|
|
q->cur++;
|
|
break;
|
|
}
|
|
break;
|
|
case '>' :
|
|
switch (q->cur[1]) {
|
|
case '>' :
|
|
switch (q->cur[2]) {
|
|
case '>' :
|
|
switch (q->cur[3]) {
|
|
case '=' :
|
|
if (q->flags & GRN_EXPR_ALLOW_UPDATE) {
|
|
PARSE(GRN_EXPR_TOKEN_SHIFTRR_ASSIGN);
|
|
q->cur += 4;
|
|
} else {
|
|
ERR(GRN_UPDATE_NOT_ALLOWED,
|
|
"'>>>=' is not allowed (%.*s)", (int)(q->str_end - q->str), q->str);
|
|
}
|
|
break;
|
|
default :
|
|
PARSE(GRN_EXPR_TOKEN_SHIFTRR);
|
|
q->cur += 3;
|
|
break;
|
|
}
|
|
break;
|
|
case '=' :
|
|
if (q->flags & GRN_EXPR_ALLOW_UPDATE) {
|
|
PARSE(GRN_EXPR_TOKEN_SHIFTR_ASSIGN);
|
|
q->cur += 3;
|
|
} else {
|
|
ERR(GRN_UPDATE_NOT_ALLOWED,
|
|
"'>>=' is not allowed (%.*s)", (int)(q->str_end - q->str), q->str);
|
|
}
|
|
break;
|
|
default :
|
|
PARSE(GRN_EXPR_TOKEN_SHIFTR);
|
|
q->cur += 2;
|
|
break;
|
|
}
|
|
break;
|
|
case '=' :
|
|
PARSE(GRN_EXPR_TOKEN_GREATER_EQUAL);
|
|
q->cur += 2;
|
|
break;
|
|
default :
|
|
PARSE(GRN_EXPR_TOKEN_GREATER);
|
|
q->cur++;
|
|
break;
|
|
}
|
|
break;
|
|
case '<' :
|
|
switch (q->cur[1]) {
|
|
case '<' :
|
|
switch (q->cur[2]) {
|
|
case '=' :
|
|
if (q->flags & GRN_EXPR_ALLOW_UPDATE) {
|
|
PARSE(GRN_EXPR_TOKEN_SHIFTL_ASSIGN);
|
|
q->cur += 3;
|
|
} else {
|
|
ERR(GRN_UPDATE_NOT_ALLOWED,
|
|
"'<<=' is not allowed (%.*s)", (int)(q->str_end - q->str), q->str);
|
|
}
|
|
break;
|
|
default :
|
|
PARSE(GRN_EXPR_TOKEN_SHIFTL);
|
|
q->cur += 2;
|
|
break;
|
|
}
|
|
break;
|
|
case '=' :
|
|
PARSE(GRN_EXPR_TOKEN_LESS_EQUAL);
|
|
q->cur += 2;
|
|
break;
|
|
default :
|
|
PARSE(GRN_EXPR_TOKEN_LESS);
|
|
q->cur++;
|
|
break;
|
|
}
|
|
break;
|
|
case '=' :
|
|
switch (q->cur[1]) {
|
|
case '=' :
|
|
PARSE(GRN_EXPR_TOKEN_EQUAL);
|
|
q->cur += 2;
|
|
break;
|
|
default :
|
|
if (q->flags & GRN_EXPR_ALLOW_UPDATE) {
|
|
PARSE(GRN_EXPR_TOKEN_ASSIGN);
|
|
q->cur++;
|
|
} else {
|
|
ERR(GRN_UPDATE_NOT_ALLOWED,
|
|
"'=' is not allowed (%.*s)", (int)(q->str_end - q->str), q->str);
|
|
}
|
|
break;
|
|
}
|
|
break;
|
|
case '0' : case '1' : case '2' : case '3' : case '4' :
|
|
case '5' : case '6' : case '7' : case '8' : case '9' :
|
|
{
|
|
const char *rest;
|
|
int64_t int64 = grn_atoll(q->cur, q->str_end, &rest);
|
|
// checks to see grn_atoll was appropriate
|
|
// (NOTE: *q->cur begins with a digit. Thus, grn_atoll parses at leaset
|
|
// one char.)
|
|
if (q->str_end != rest &&
|
|
(*rest == '.' || *rest == 'e' || *rest == 'E' ||
|
|
(*rest >= '0' && *rest <= '9'))) {
|
|
char *rest_float;
|
|
double d = strtod(q->cur, &rest_float);
|
|
grn_obj floatbuf;
|
|
GRN_FLOAT_INIT(&floatbuf, 0);
|
|
GRN_FLOAT_SET(ctx, &floatbuf, d);
|
|
grn_expr_append_const(ctx, q->e, &floatbuf, GRN_OP_PUSH, 1);
|
|
rest = rest_float;
|
|
} else {
|
|
const char *rest64 = rest;
|
|
grn_atoui(q->cur, q->str_end, &rest);
|
|
// checks to see grn_atoi failed (see above NOTE)
|
|
if ((int64 > UINT32_MAX) ||
|
|
(q->str_end != rest && *rest >= '0' && *rest <= '9')) {
|
|
grn_obj int64buf;
|
|
GRN_INT64_INIT(&int64buf, 0);
|
|
GRN_INT64_SET(ctx, &int64buf, int64);
|
|
grn_expr_append_const(ctx, q->e, &int64buf, GRN_OP_PUSH, 1);
|
|
rest = rest64;
|
|
} else if (int64 > INT32_MAX || int64 < INT32_MIN) {
|
|
grn_obj int64buf;
|
|
GRN_INT64_INIT(&int64buf, 0);
|
|
GRN_INT64_SET(ctx, &int64buf, int64);
|
|
grn_expr_append_const(ctx, q->e, &int64buf, GRN_OP_PUSH, 1);
|
|
} else {
|
|
grn_obj int32buf;
|
|
GRN_INT32_INIT(&int32buf, 0);
|
|
GRN_INT32_SET(ctx, &int32buf, (int32_t)int64);
|
|
grn_expr_append_const(ctx, q->e, &int32buf, GRN_OP_PUSH, 1);
|
|
}
|
|
}
|
|
PARSE(GRN_EXPR_TOKEN_DECIMAL);
|
|
q->cur = rest;
|
|
}
|
|
break;
|
|
default :
|
|
if ((rc = get_identifier(ctx, q))) { goto exit; }
|
|
break;
|
|
}
|
|
if (ctx->rc) { rc = ctx->rc; break; }
|
|
}
|
|
exit :
|
|
PARSE(0);
|
|
return rc;
|
|
}
|
|
|
|
grn_rc
|
|
grn_expr_parse(grn_ctx *ctx, grn_obj *expr,
|
|
const char *str, unsigned int str_size,
|
|
grn_obj *default_column, grn_operator default_mode,
|
|
grn_operator default_op, grn_expr_flags flags)
|
|
{
|
|
efs_info efsi;
|
|
if (grn_expr_parser_open(ctx)) { return ctx->rc; }
|
|
GRN_API_ENTER;
|
|
efsi.ctx = ctx;
|
|
efsi.str = str;
|
|
if ((efsi.v = grn_expr_get_var_by_offset(ctx, expr, 0)) &&
|
|
(efsi.table = grn_ctx_at(ctx, efsi.v->header.domain))) {
|
|
GRN_TEXT_INIT(&efsi.buf, 0);
|
|
GRN_INT32_INIT(&efsi.op_stack, GRN_OBJ_VECTOR);
|
|
GRN_INT32_INIT(&efsi.mode_stack, GRN_OBJ_VECTOR);
|
|
GRN_INT32_INIT(&efsi.max_interval_stack, GRN_OBJ_VECTOR);
|
|
GRN_INT32_INIT(&efsi.similarity_threshold_stack, GRN_OBJ_VECTOR);
|
|
GRN_PTR_INIT(&efsi.column_stack, GRN_OBJ_VECTOR, GRN_ID_NIL);
|
|
GRN_PTR_INIT(&efsi.token_stack, GRN_OBJ_VECTOR, GRN_ID_NIL);
|
|
efsi.e = expr;
|
|
efsi.str = str;
|
|
efsi.cur = str;
|
|
efsi.str_end = str + str_size;
|
|
efsi.default_column = default_column;
|
|
GRN_PTR_PUT(ctx, &efsi.column_stack, default_column);
|
|
GRN_INT32_PUT(ctx, &efsi.op_stack, default_op);
|
|
GRN_INT32_PUT(ctx, &efsi.mode_stack, default_mode);
|
|
efsi.default_flags = efsi.flags = flags;
|
|
efsi.escalation_threshold = GRN_DEFAULT_MATCH_ESCALATION_THRESHOLD;
|
|
efsi.escalation_decaystep = DEFAULT_DECAYSTEP;
|
|
efsi.weight_offset = 0;
|
|
efsi.opt.weight_vector = NULL;
|
|
efsi.weight_set = NULL;
|
|
|
|
if (flags & (GRN_EXPR_SYNTAX_SCRIPT |
|
|
GRN_EXPR_SYNTAX_OUTPUT_COLUMNS |
|
|
GRN_EXPR_SYNTAX_ADJUSTER)) {
|
|
efs_info *q = &efsi;
|
|
if (flags & GRN_EXPR_SYNTAX_OUTPUT_COLUMNS) {
|
|
PARSE(GRN_EXPR_TOKEN_START_OUTPUT_COLUMNS);
|
|
} else if (flags & GRN_EXPR_SYNTAX_ADJUSTER) {
|
|
PARSE(GRN_EXPR_TOKEN_START_ADJUSTER);
|
|
}
|
|
parse_script(ctx, &efsi);
|
|
} else {
|
|
parse_query(ctx, &efsi);
|
|
}
|
|
|
|
/*
|
|
grn_obj strbuf;
|
|
GRN_TEXT_INIT(&strbuf, 0);
|
|
grn_expr_inspect_internal(ctx, &strbuf, expr);
|
|
GRN_TEXT_PUTC(ctx, &strbuf, '\0');
|
|
GRN_LOG(ctx, GRN_LOG_NOTICE, "query=(%s)", GRN_TEXT_VALUE(&strbuf));
|
|
GRN_OBJ_FIN(ctx, &strbuf);
|
|
*/
|
|
|
|
/*
|
|
efsi.opt.vector_size = DEFAULT_WEIGHT_VECTOR_SIZE;
|
|
efsi.opt.func = efsi.weight_set ? section_weight_cb : NULL;
|
|
efsi.opt.func_arg = efsi.weight_set;
|
|
efsi.snip_conds = NULL;
|
|
*/
|
|
GRN_OBJ_FIN(ctx, &efsi.op_stack);
|
|
GRN_OBJ_FIN(ctx, &efsi.mode_stack);
|
|
GRN_OBJ_FIN(ctx, &efsi.max_interval_stack);
|
|
GRN_OBJ_FIN(ctx, &efsi.similarity_threshold_stack);
|
|
GRN_OBJ_FIN(ctx, &efsi.column_stack);
|
|
GRN_OBJ_FIN(ctx, &efsi.token_stack);
|
|
GRN_OBJ_FIN(ctx, &efsi.buf);
|
|
} else {
|
|
ERR(GRN_INVALID_ARGUMENT, "variable is not defined correctly");
|
|
}
|
|
GRN_API_RETURN(ctx->rc);
|
|
}
|
|
|
|
grn_rc
|
|
grn_expr_parser_close(grn_ctx *ctx)
|
|
{
|
|
if (ctx->impl->parser) {
|
|
yyParser *pParser = (yyParser*)ctx->impl->parser;
|
|
while (pParser->yyidx >= 0) yy_pop_parser_stack(pParser);
|
|
#if YYSTACKDEPTH<=0
|
|
free(pParser->yystack);
|
|
#endif
|
|
GRN_FREE(pParser);
|
|
ctx->impl->parser = NULL;
|
|
}
|
|
return ctx->rc;
|
|
}
|
|
|
|
grn_rc
|
|
grn_expr_get_keywords(grn_ctx *ctx, grn_obj *expr, grn_obj *keywords)
|
|
{
|
|
int i, n;
|
|
scan_info **sis, *si;
|
|
GRN_API_ENTER;
|
|
if ((sis = scan_info_build(ctx, expr, &n, GRN_OP_OR, 0))) {
|
|
int butp = 0, nparens = 0, npbut = 0;
|
|
grn_obj but_stack;
|
|
GRN_UINT32_INIT(&but_stack, GRN_OBJ_VECTOR);
|
|
for (i = n; i--;) {
|
|
si = sis[i];
|
|
if (si->flags & SCAN_POP) {
|
|
nparens++;
|
|
if (si->logical_op == GRN_OP_AND_NOT) {
|
|
GRN_UINT32_PUT(ctx, &but_stack, npbut);
|
|
npbut = nparens;
|
|
butp = 1 - butp;
|
|
}
|
|
} else {
|
|
if (si->op == GRN_OP_MATCH && si->query) {
|
|
if (butp == (si->logical_op == GRN_OP_AND_NOT)) {
|
|
GRN_PTR_PUT(ctx, keywords, si->query);
|
|
}
|
|
}
|
|
if (si->flags & SCAN_PUSH) {
|
|
if (nparens == npbut) {
|
|
butp = 1 - butp;
|
|
GRN_UINT32_POP(&but_stack, npbut);
|
|
}
|
|
nparens--;
|
|
}
|
|
}
|
|
}
|
|
GRN_OBJ_FIN(ctx, &but_stack);
|
|
for (i = n; i--;) { SI_FREE(sis[i]); }
|
|
GRN_FREE(sis);
|
|
}
|
|
GRN_API_RETURN(GRN_SUCCESS);
|
|
}
|
|
|
|
grn_rc
|
|
grn_expr_snip_add_conditions(grn_ctx *ctx, grn_obj *expr, grn_obj *snip,
|
|
unsigned int n_tags,
|
|
const char **opentags, unsigned int *opentag_lens,
|
|
const char **closetags, unsigned int *closetag_lens)
|
|
{
|
|
grn_rc rc;
|
|
grn_obj keywords;
|
|
|
|
GRN_API_ENTER;
|
|
|
|
GRN_PTR_INIT(&keywords, GRN_OBJ_VECTOR, GRN_ID_NIL);
|
|
rc = grn_expr_get_keywords(ctx, expr, &keywords);
|
|
if (rc != GRN_SUCCESS) {
|
|
GRN_OBJ_FIN(ctx, &keywords);
|
|
GRN_API_RETURN(rc);
|
|
}
|
|
|
|
if (n_tags) {
|
|
int i;
|
|
for (i = 0;; i = (i + 1) % n_tags) {
|
|
grn_obj *keyword;
|
|
GRN_PTR_POP(&keywords, keyword);
|
|
if (!keyword) { break; }
|
|
grn_snip_add_cond(ctx, snip,
|
|
GRN_TEXT_VALUE(keyword), GRN_TEXT_LEN(keyword),
|
|
opentags[i], opentag_lens[i],
|
|
closetags[i], closetag_lens[i]);
|
|
}
|
|
} else {
|
|
for (;;) {
|
|
grn_obj *keyword;
|
|
GRN_PTR_POP(&keywords, keyword);
|
|
if (!keyword) { break; }
|
|
grn_snip_add_cond(ctx, snip,
|
|
GRN_TEXT_VALUE(keyword), GRN_TEXT_LEN(keyword),
|
|
NULL, 0, NULL, 0);
|
|
}
|
|
}
|
|
GRN_OBJ_FIN(ctx, &keywords);
|
|
|
|
GRN_API_RETURN(GRN_SUCCESS);
|
|
}
|
|
|
|
grn_obj *
|
|
grn_expr_snip(grn_ctx *ctx, grn_obj *expr, int flags,
|
|
unsigned int width, unsigned int max_results,
|
|
unsigned int n_tags,
|
|
const char **opentags, unsigned int *opentag_lens,
|
|
const char **closetags, unsigned int *closetag_lens,
|
|
grn_snip_mapping *mapping)
|
|
{
|
|
grn_obj *res = NULL;
|
|
GRN_API_ENTER;
|
|
if ((res = grn_snip_open(ctx, flags, width, max_results,
|
|
NULL, 0, NULL, 0, mapping))) {
|
|
grn_expr_snip_add_conditions(ctx, expr, res,
|
|
n_tags,
|
|
opentags, opentag_lens,
|
|
closetags, closetag_lens);
|
|
}
|
|
GRN_API_RETURN(res);
|
|
}
|
|
|
|
/*
|
|
So far, grn_column_filter() is nothing but a very rough prototype.
|
|
Although GRN_COLUMN_EACH() can accelerate many range queries,
|
|
the following stuff must be resolved one by one.
|
|
|
|
* support accessors as column
|
|
* support tables which have deleted records
|
|
* support various operators
|
|
* support various column types
|
|
*/
|
|
grn_rc
|
|
grn_column_filter(grn_ctx *ctx, grn_obj *column,
|
|
grn_operator operator,
|
|
grn_obj *value, grn_obj *result_set,
|
|
grn_operator set_operation)
|
|
{
|
|
uint32_t *vp;
|
|
grn_ii_posting posting;
|
|
uint32_t value_ = grn_atoui(GRN_TEXT_VALUE(value), GRN_BULK_CURR(value), NULL);
|
|
posting.sid = 1;
|
|
posting.pos = 0;
|
|
posting.weight = 0;
|
|
GRN_COLUMN_EACH(ctx, column, id, vp, {
|
|
if (*vp < value_) {
|
|
posting.rid = id;
|
|
grn_ii_posting_add(ctx, &posting, (grn_hash *)result_set, set_operation);
|
|
}
|
|
});
|
|
grn_ii_resolve_sel_and(ctx, (grn_hash *)result_set, set_operation);
|
|
return ctx->rc;
|
|
}
|
|
|
|
grn_rc
|
|
grn_expr_syntax_escape(grn_ctx *ctx, const char *string, int string_size,
|
|
const char *target_characters,
|
|
char escape_character,
|
|
grn_obj *escaped_string)
|
|
{
|
|
grn_rc rc = GRN_SUCCESS;
|
|
const char *current, *string_end;
|
|
|
|
if (!string) {
|
|
return GRN_INVALID_ARGUMENT;
|
|
}
|
|
|
|
GRN_API_ENTER;
|
|
if (string_size < 0) {
|
|
string_size = strlen(string);
|
|
}
|
|
string_end = string + string_size;
|
|
|
|
current = string;
|
|
while (current < string_end) {
|
|
unsigned int char_size;
|
|
char_size = grn_charlen(ctx, current, string_end);
|
|
switch (char_size) {
|
|
case 0 :
|
|
/* string includes malformed multibyte character. */
|
|
return GRN_INVALID_ARGUMENT;
|
|
break;
|
|
case 1 :
|
|
if (strchr(target_characters, *current)) {
|
|
GRN_TEXT_PUTC(ctx, escaped_string, escape_character);
|
|
}
|
|
GRN_TEXT_PUT(ctx, escaped_string, current, char_size);
|
|
current += char_size;
|
|
break;
|
|
default :
|
|
GRN_TEXT_PUT(ctx, escaped_string, current, char_size);
|
|
current += char_size;
|
|
break;
|
|
}
|
|
}
|
|
|
|
GRN_API_RETURN(rc);
|
|
}
|
|
|
|
grn_rc
|
|
grn_expr_syntax_escape_query(grn_ctx *ctx, const char *query, int query_size,
|
|
grn_obj *escaped_query)
|
|
{
|
|
const char target_characters[] = {
|
|
GRN_QUERY_AND,
|
|
GRN_QUERY_AND_NOT,
|
|
GRN_QUERY_ADJ_INC,
|
|
GRN_QUERY_ADJ_DEC,
|
|
GRN_QUERY_ADJ_NEG,
|
|
GRN_QUERY_PREFIX,
|
|
GRN_QUERY_PARENL,
|
|
GRN_QUERY_PARENR,
|
|
GRN_QUERY_QUOTEL,
|
|
GRN_QUERY_ESCAPE,
|
|
GRN_QUERY_COLUMN,
|
|
'\0',
|
|
};
|
|
return grn_expr_syntax_escape(ctx, query, query_size,
|
|
target_characters, GRN_QUERY_ESCAPE,
|
|
escaped_query);
|
|
}
|
|
|
|
grn_rc
|
|
grn_expr_dump_plan(grn_ctx *ctx, grn_obj *expr, grn_obj *buffer)
|
|
{
|
|
int n;
|
|
scan_info **sis;
|
|
|
|
GRN_API_ENTER;
|
|
sis = scan_info_build(ctx, expr, &n, GRN_OP_OR, 0);
|
|
if (sis) {
|
|
int i;
|
|
grn_inspect_scan_info_list(ctx, buffer, sis, n);
|
|
for (i = 0; i < n; i++) {
|
|
SI_FREE(sis[i]);
|
|
}
|
|
GRN_FREE(sis);
|
|
} else {
|
|
GRN_TEXT_PUTS(ctx, buffer, "sequential search\n");
|
|
}
|
|
GRN_API_RETURN(GRN_SUCCESS);
|
|
}
|