2010-12-14 13:41:37 -03:00
|
|
|
%{
|
|
|
|
/*
|
|
|
|
* lang.y - Toolpath adaptation language
|
|
|
|
*
|
|
|
|
* Written 2010 by Werner Almesberger
|
|
|
|
* Copyright 2010 by Werner Almesberger
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
|
|
|
#include "path.h"
|
2010-12-14 14:20:19 -03:00
|
|
|
#include "ops.h"
|
2010-12-14 13:41:37 -03:00
|
|
|
#include "gnuplot.h"
|
|
|
|
#include "gerber.h"
|
2010-12-15 03:27:55 -03:00
|
|
|
#include "excellon.h"
|
2010-12-14 13:41:37 -03:00
|
|
|
|
|
|
|
#include "y.tab.h"
|
|
|
|
|
|
|
|
|
|
|
|
static double xo = 0, yo = 0, zo = 0;
|
|
|
|
static struct path *paths = NULL;
|
2010-12-15 04:56:58 -03:00
|
|
|
static struct path *remain = NULL;
|
2010-12-14 13:41:37 -03:00
|
|
|
|
|
|
|
|
2010-12-14 19:50:53 -03:00
|
|
|
#define MIL2MM(mil) ((mil)/1000*25.4)
|
|
|
|
|
|
|
|
|
2010-12-14 13:41:37 -03:00
|
|
|
static void add_paths(struct path *new)
|
|
|
|
{
|
|
|
|
struct path **anchor = &paths;
|
|
|
|
|
|
|
|
while (*anchor)
|
|
|
|
anchor = &(*anchor)->next;
|
|
|
|
*anchor = new;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-12-15 06:34:53 -03:00
|
|
|
static void translate(struct path *list, double x, double y, double z)
|
2010-12-14 13:41:37 -03:00
|
|
|
{
|
|
|
|
struct path *path;
|
|
|
|
struct point *p;
|
|
|
|
|
2010-12-15 06:34:53 -03:00
|
|
|
for (path = list; path; path = path->next)
|
2010-12-14 13:41:37 -03:00
|
|
|
for (p = path->first; p; p = p->next) {
|
|
|
|
p->x += x;
|
|
|
|
p->y += y;
|
|
|
|
p->z += z;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static double ref_pick_1(int ref, double a, double b)
|
|
|
|
{
|
|
|
|
switch (ref) {
|
|
|
|
case 0:
|
|
|
|
return a;
|
|
|
|
case 1:
|
|
|
|
return (a+b)/2;
|
|
|
|
case 2:
|
|
|
|
return b;
|
|
|
|
default:
|
|
|
|
abort();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void ref_pick_xy(int ref, double xa, double ya, double xb, double yb,
|
|
|
|
double *x, double *y)
|
|
|
|
{
|
|
|
|
*x = ref_pick_1((ref-1) % 3, xa, xb);
|
|
|
|
*y = ref_pick_1((ref-1)/3, ya, yb);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void align(int ref, double x, double y)
|
|
|
|
{
|
2010-12-14 14:04:35 -03:00
|
|
|
const struct path *path;
|
|
|
|
const struct point *p;
|
|
|
|
double xa = 0, ya = 0, xb = 0, yb = 0;
|
|
|
|
double xr, yr, xd, yd;
|
|
|
|
int first = 1;
|
|
|
|
|
|
|
|
for (path = paths; path; path = path->next)
|
|
|
|
for (p = path->first; p; p = p->next) {
|
|
|
|
if (first || p->x < xa)
|
|
|
|
xa = p->x;
|
|
|
|
if (first || p->x > xb)
|
|
|
|
xb = p->x;
|
|
|
|
if (first || p->y < ya)
|
|
|
|
ya = p->y;
|
|
|
|
if (first || p->y > yb)
|
|
|
|
yb = p->y;
|
|
|
|
first = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
ref_pick_xy(ref, xa, ya, xb, yb, &xr, &yr);
|
|
|
|
xd = x-xr;
|
|
|
|
yd = y-yr;
|
|
|
|
|
2010-12-15 06:34:53 -03:00
|
|
|
translate(paths, xd, yd, 0);
|
2010-12-14 14:04:35 -03:00
|
|
|
xo += xd;
|
|
|
|
yo += yd;
|
2010-12-14 13:41:37 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-12-15 04:56:58 -03:00
|
|
|
static void clear_paths(void)
|
|
|
|
{
|
|
|
|
struct path *next;
|
|
|
|
|
|
|
|
while (paths) {
|
|
|
|
next = paths->next;
|
|
|
|
path_free(paths);
|
|
|
|
paths = next;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static struct path **classify(struct path **anchor, struct path *path)
|
|
|
|
{
|
|
|
|
struct path **walk, *next;
|
|
|
|
|
|
|
|
if (!path)
|
|
|
|
return &(*anchor)->next;
|
|
|
|
for (walk = &paths; *walk; walk = &(*walk)->next);
|
|
|
|
*walk = path;
|
|
|
|
next = (*anchor)->next;
|
|
|
|
path_free(*anchor);
|
|
|
|
*anchor = next;
|
|
|
|
return anchor;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-12-14 13:41:37 -03:00
|
|
|
%}
|
|
|
|
|
|
|
|
|
|
|
|
%union {
|
|
|
|
double num;
|
|
|
|
char *str;
|
2010-12-14 23:37:31 -03:00
|
|
|
enum {
|
|
|
|
OO_DOG = 1 << 0,
|
|
|
|
OO_INSIDE = 1 << 1,
|
|
|
|
} oopt;
|
2010-12-14 13:41:37 -03:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2010-12-15 04:56:58 -03:00
|
|
|
%token TOK_ALIGN TOK_ARRAY TOK_CLEAR TOK_DRILL TOK_EMPTY
|
|
|
|
%token TOK_MILL TOK_OFFSET TOK_REMAINDER TOK_RESET
|
2010-12-14 13:41:37 -03:00
|
|
|
%token TOK_TRANSLATE TOK_Z
|
2010-12-15 06:27:15 -03:00
|
|
|
%token TOK_APPEND TOK_GERBER TOK_GNUPLOT TOK_EXCELLON TOK_WRITE
|
2010-12-14 23:37:31 -03:00
|
|
|
%token TOK_DOG TOK_INSIDE
|
2010-12-14 13:41:37 -03:00
|
|
|
|
2010-12-14 19:50:53 -03:00
|
|
|
%token <num> NUM_EXP_MIL NUM_EXP_MM NUM_IMP_MIL NUM_IMP_MM REF
|
2010-12-14 13:41:37 -03:00
|
|
|
%token <str> STRING
|
|
|
|
|
|
|
|
%type <str> opt_filename
|
2010-12-14 19:50:53 -03:00
|
|
|
%type <num> dimen number
|
2010-12-14 23:37:31 -03:00
|
|
|
%type <oopt> offset_options offset_option
|
2010-12-14 13:41:37 -03:00
|
|
|
|
|
|
|
%%
|
|
|
|
|
|
|
|
all:
|
|
|
|
| command all
|
|
|
|
;
|
|
|
|
|
|
|
|
command:
|
2010-12-14 19:50:53 -03:00
|
|
|
TOK_ALIGN REF dimen dimen
|
2010-12-14 13:41:37 -03:00
|
|
|
{
|
|
|
|
align((int) $2, $3, $4);
|
|
|
|
}
|
2010-12-14 19:50:53 -03:00
|
|
|
| TOK_ALIGN REF dimen dimen dimen dimen
|
2010-12-14 13:41:37 -03:00
|
|
|
{
|
|
|
|
int ref = $2;
|
|
|
|
double x, y;
|
|
|
|
|
2010-12-14 19:17:34 -03:00
|
|
|
if ($3 > $5)
|
|
|
|
yyerror("left edge > right edge");
|
|
|
|
if ($4 > $6)
|
|
|
|
yyerror("bottom > top");
|
|
|
|
|
2010-12-14 13:41:37 -03:00
|
|
|
ref_pick_xy(ref, $3, $4, $5, $6, &x, &y);
|
|
|
|
align(ref, x, y);
|
|
|
|
}
|
2010-12-14 19:50:53 -03:00
|
|
|
| TOK_ARRAY dimen dimen number number
|
2010-12-14 13:41:37 -03:00
|
|
|
{
|
|
|
|
double x = $2*$4;
|
|
|
|
double y = $3*$5;
|
|
|
|
|
2010-12-15 06:34:53 -03:00
|
|
|
translate(paths, x, y, 0);
|
2010-12-15 06:30:05 -03:00
|
|
|
xo += x;
|
|
|
|
yo += y;
|
2010-12-14 13:41:37 -03:00
|
|
|
}
|
|
|
|
| TOK_CLEAR
|
|
|
|
{
|
2010-12-15 04:56:58 -03:00
|
|
|
clear_paths();
|
2010-12-14 13:41:37 -03:00
|
|
|
}
|
|
|
|
| TOK_RESET
|
|
|
|
{
|
|
|
|
xo = yo = 0;
|
|
|
|
}
|
2010-12-14 23:37:31 -03:00
|
|
|
| TOK_OFFSET offset_options
|
2010-12-14 13:41:37 -03:00
|
|
|
{
|
2010-12-14 23:37:31 -03:00
|
|
|
tool_comp_paths(paths,
|
|
|
|
!!($2 & OO_DOG), !!($2 & OO_INSIDE));
|
2010-12-14 13:41:37 -03:00
|
|
|
}
|
2010-12-14 19:50:53 -03:00
|
|
|
| TOK_TRANSLATE dimen dimen
|
2010-12-14 13:41:37 -03:00
|
|
|
{
|
2010-12-15 06:34:53 -03:00
|
|
|
translate(paths, $2, $3, 0);
|
2010-12-14 13:41:37 -03:00
|
|
|
xo += $2;
|
|
|
|
yo += $3;
|
|
|
|
}
|
2010-12-14 19:50:53 -03:00
|
|
|
| TOK_Z dimen
|
2010-12-14 13:41:37 -03:00
|
|
|
{
|
2010-12-14 16:49:26 -03:00
|
|
|
zo += $2;
|
2010-12-14 13:41:37 -03:00
|
|
|
}
|
2010-12-14 19:50:53 -03:00
|
|
|
| TOK_Z dimen dimen
|
2010-12-14 13:41:37 -03:00
|
|
|
{
|
2010-12-14 16:49:26 -03:00
|
|
|
zo += $3-$2;
|
2010-12-14 13:41:37 -03:00
|
|
|
}
|
2010-12-14 19:50:53 -03:00
|
|
|
| TOK_GERBER dimen opt_filename
|
2010-12-14 13:41:37 -03:00
|
|
|
{
|
2010-12-15 06:34:53 -03:00
|
|
|
struct path *new;
|
|
|
|
|
|
|
|
new = gerber_read($3, $2/2);
|
|
|
|
translate(new, xo, yo, 0);
|
|
|
|
add_paths(new);
|
2010-12-14 13:41:37 -03:00
|
|
|
}
|
2010-12-14 19:50:53 -03:00
|
|
|
| TOK_GNUPLOT dimen opt_filename
|
2010-12-14 13:41:37 -03:00
|
|
|
{
|
2010-12-15 06:34:53 -03:00
|
|
|
struct path *new;
|
|
|
|
|
|
|
|
new = gnuplot_read($3, $2/2);
|
|
|
|
translate(new, xo, yo, 0);
|
|
|
|
add_paths(new);
|
2010-12-14 13:41:37 -03:00
|
|
|
}
|
2010-12-15 03:27:55 -03:00
|
|
|
| TOK_EXCELLON opt_filename
|
|
|
|
{
|
2010-12-15 06:34:53 -03:00
|
|
|
struct path *new;
|
|
|
|
|
|
|
|
new = excellon_read($2);
|
|
|
|
translate(new, xo, yo, 0);
|
|
|
|
add_paths(new);
|
2010-12-15 03:27:55 -03:00
|
|
|
}
|
2010-12-14 13:41:37 -03:00
|
|
|
| TOK_WRITE opt_filename
|
|
|
|
{
|
2010-12-15 06:34:53 -03:00
|
|
|
translate(paths, 0, 0, zo);
|
2010-12-14 13:41:37 -03:00
|
|
|
gnuplot_write($2, paths);
|
2010-12-15 06:34:53 -03:00
|
|
|
translate(paths, 0, 0, -zo);
|
2010-12-14 13:41:37 -03:00
|
|
|
}
|
2010-12-15 06:27:15 -03:00
|
|
|
| TOK_APPEND opt_filename
|
|
|
|
{
|
2010-12-15 06:34:53 -03:00
|
|
|
translate(paths, 0, 0, zo);
|
2010-12-15 06:27:15 -03:00
|
|
|
gnuplot_append($2, paths);
|
2010-12-15 06:34:53 -03:00
|
|
|
translate(paths, 0, 0, -zo);
|
2010-12-15 06:27:15 -03:00
|
|
|
}
|
2010-12-15 04:56:58 -03:00
|
|
|
| TOK_DRILL dimen dimen
|
|
|
|
{
|
|
|
|
struct path **walk;
|
|
|
|
|
|
|
|
remain = paths;
|
|
|
|
paths = NULL;
|
2010-12-15 06:13:40 -03:00
|
|
|
walk = &remain;
|
2010-12-15 04:56:58 -03:00
|
|
|
while (*walk)
|
|
|
|
walk =
|
|
|
|
classify(walk, try_drill(*walk, $2, $3));
|
|
|
|
}
|
|
|
|
| TOK_MILL dimen dimen
|
|
|
|
{
|
|
|
|
struct path **walk;
|
|
|
|
|
|
|
|
remain = paths;
|
|
|
|
paths = NULL;
|
2010-12-15 06:13:40 -03:00
|
|
|
walk = &remain;
|
2010-12-15 04:56:58 -03:00
|
|
|
while (*walk)
|
|
|
|
walk = classify(walk, try_mill(*walk, $2, $3));
|
|
|
|
}
|
|
|
|
| TOK_REMAINDER
|
|
|
|
{
|
|
|
|
clear_paths();
|
|
|
|
paths = remain;
|
|
|
|
remain = NULL;
|
|
|
|
}
|
|
|
|
| TOK_EMPTY
|
|
|
|
{
|
|
|
|
if (paths)
|
|
|
|
yyerror("path list is not empty");
|
|
|
|
}
|
2010-12-14 13:41:37 -03:00
|
|
|
;
|
|
|
|
|
|
|
|
opt_filename:
|
|
|
|
{
|
|
|
|
$$ = NULL;
|
|
|
|
}
|
|
|
|
| STRING
|
|
|
|
{
|
|
|
|
$$ = $1;
|
|
|
|
}
|
|
|
|
;
|
2010-12-14 19:50:53 -03:00
|
|
|
|
|
|
|
dimen:
|
|
|
|
NUM_EXP_MM
|
|
|
|
{
|
|
|
|
$$ = $1;
|
|
|
|
}
|
|
|
|
| NUM_IMP_MM
|
|
|
|
{
|
|
|
|
$$ = $1;
|
|
|
|
}
|
|
|
|
| NUM_EXP_MIL
|
|
|
|
{
|
|
|
|
$$ = MIL2MM($1);
|
|
|
|
}
|
|
|
|
| NUM_IMP_MIL
|
|
|
|
{
|
|
|
|
$$ = MIL2MM($1);
|
|
|
|
}
|
|
|
|
;
|
|
|
|
|
|
|
|
number:
|
|
|
|
NUM_IMP_MIL
|
|
|
|
{
|
|
|
|
$$ = $1;
|
|
|
|
}
|
|
|
|
| NUM_IMP_MM
|
|
|
|
{
|
|
|
|
$$ = $1;
|
|
|
|
}
|
|
|
|
;
|
2010-12-14 23:37:31 -03:00
|
|
|
|
|
|
|
offset_options:
|
|
|
|
{
|
|
|
|
$$ = 0;
|
|
|
|
}
|
|
|
|
| offset_option offset_options
|
|
|
|
{
|
|
|
|
$$ = $1 | $2;
|
|
|
|
|
|
|
|
}
|
|
|
|
;
|
|
|
|
|
|
|
|
offset_option:
|
|
|
|
TOK_DOG
|
|
|
|
{
|
|
|
|
$$ = OO_DOG;
|
|
|
|
}
|
|
|
|
| TOK_INSIDE
|
|
|
|
{
|
|
|
|
$$ = OO_INSIDE;
|
|
|
|
}
|
|
|
|
;
|