mirror of
https://github.com/ra3xdh/qucs_s
synced 2025-03-28 21:13:26 +00:00
502 lines
22 KiB
C++
502 lines
22 KiB
C++
/***************************************************************************
|
|
sallenkey.cpp
|
|
----------------
|
|
begin : Wed Apr 10 2014
|
|
copyright : (C) 2014 by Vadim Kuznetsov
|
|
email : ra3xdh@gmail.com
|
|
***************************************************************************/
|
|
|
|
/***************************************************************************
|
|
* *
|
|
* 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. *
|
|
* *
|
|
***************************************************************************/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
# include <config.h>
|
|
#endif
|
|
|
|
#include "sallenkey.h"
|
|
#include <iostream>
|
|
|
|
SallenKey::SallenKey(Filter::FilterFunc ffunc_, Filter::FType type_, FilterParam par) :
|
|
Filter(ffunc_, type_, par)
|
|
{
|
|
if ((ftype==Filter::LowPass)||(ftype==Filter::BandPass)) Nr1 = 4;
|
|
if ((ftype==Filter::BandPass)||(ftype==Filter::BandStop)) Nr1 = 5;
|
|
|
|
Nc1 = 2;
|
|
Nop1 = 1;
|
|
}
|
|
|
|
|
|
void SallenKey::calcLowPass()
|
|
{
|
|
double R1,R2,R3,R4,C1,C2;
|
|
double Wc = 2*pi*Fc;
|
|
double Nst = order/2 + order%2;
|
|
double Kv1 = pow(Kv,1.0/Nst);
|
|
|
|
for (int k=1; k <= order/2; k++) {
|
|
|
|
double re = Poles.at(k-1).real();
|
|
double im = Poles.at(k-1).imag();
|
|
double B = -2.0*re;
|
|
double C = re*re + im*im;
|
|
|
|
C2 = 10 / Fc;
|
|
C1 = (B*B+4*C*(Kv1-1))*C2/(4*C);
|
|
R1 = 2/(Wc*(B*C2+sqrt((B*B + 4*C*(Kv1-1))*C2*C2-4*C*C1*C2)));
|
|
R2 = 1/(C*C1*C2*R1*Wc*Wc);
|
|
|
|
if (Kv != 1.0) {
|
|
R3 = Kv1*(R1 + R2)/(Kv1 - 1);
|
|
R4 = Kv1*(R1 + R2);
|
|
} else {
|
|
R3 = 1;
|
|
R4 = 0;
|
|
}
|
|
|
|
RC_elements curr_stage;
|
|
curr_stage.N = k;
|
|
curr_stage.R1 = 1000*R1;
|
|
curr_stage.R2 = 1000*R2;
|
|
curr_stage.R3 = 1000*R3;
|
|
curr_stage.R4 = 1000*R4;
|
|
curr_stage.R5 = 0;
|
|
curr_stage.R6 = 0;
|
|
curr_stage.C1 = C1;
|
|
curr_stage.C2 = C2;
|
|
Sections.append(curr_stage);
|
|
}
|
|
|
|
this->calcFirstOrder();
|
|
}
|
|
|
|
|
|
void SallenKey::calcHighPass()
|
|
{
|
|
double R1,R2,R3,R4,C1;
|
|
double Wc = 2*pi*Fc;
|
|
|
|
double Nst = order/2 + order%2;
|
|
double Kv1 = pow(Kv,1.0/Nst);
|
|
|
|
for (int k=1; k <= order/2; k++) {
|
|
|
|
double re = Poles.at(k-1).real();
|
|
double im = Poles.at(k-1).imag();
|
|
double B = -2.0*re;
|
|
double C = re*re + im*im;
|
|
|
|
C1 = 10 / Fc;
|
|
|
|
R2 = 4*C/(Wc*C1*(B+sqrt(B*B+8*C*(Kv1-1))));
|
|
|
|
R1 = C/(Wc*Wc*C1*C1*R2);
|
|
|
|
if (Kv != 1.0) {
|
|
R3 = Kv1*R2/(Kv1 - 1);
|
|
R4 = Kv1*R2;
|
|
} else {
|
|
R3 = 1;
|
|
R4 = 0;
|
|
}
|
|
|
|
RC_elements curr_stage;
|
|
curr_stage.N = k;
|
|
curr_stage.R1 = 1000*R1;
|
|
curr_stage.R2 = 1000*R2;
|
|
curr_stage.R3 = 1000*R3;
|
|
curr_stage.R4 = 1000*R4;
|
|
curr_stage.R5 = 0;
|
|
curr_stage.R6 = 0;
|
|
curr_stage.C1 = C1;
|
|
curr_stage.C2 = C1;
|
|
Sections.append(curr_stage);
|
|
}
|
|
calcFirstOrder();
|
|
}
|
|
|
|
|
|
void SallenKey::calcBandPass()
|
|
{
|
|
double W0 = 2*pi*F0;
|
|
double R1,R2,R3,R4,C1;
|
|
//float rho = Kv/Q;
|
|
//float gamma = 1.0;
|
|
int cnt = 1;
|
|
double Kv1 = pow(Kv,1.0/order);
|
|
|
|
if (order==1) { // Filter contains only 1 1st-order section
|
|
double rho = Kv1/Q;
|
|
double beta = 1.0/Q;
|
|
double gamma = 1.0;
|
|
|
|
C1 = 10.0/F0;
|
|
R1 = 2.0/(rho*W0*C1);
|
|
R2 = 2.0/((-beta+sqrt((rho-beta)*(rho-beta)+8.0*gamma))*W0*C1);
|
|
R3 = (1.0/R1+1.0/R2)/(gamma*W0*W0*C1*C1);
|
|
R4 = 2.0*R3;
|
|
|
|
RC_elements current_section;
|
|
current_section.N = 1;
|
|
current_section.R1 = 1000*R1;
|
|
current_section.R2 = 1000*R2;
|
|
current_section.R3 = 1000*R3;
|
|
current_section.R4 = 1000*R4;
|
|
current_section.R5 = 0;
|
|
current_section.R6 = 0;
|
|
current_section.C1 = C1;
|
|
current_section.C2 = C1;
|
|
Sections.append(current_section);
|
|
|
|
return;
|
|
}
|
|
|
|
for (int k=1; k <= order/2; k++) { // Usually 2nd-order section
|
|
double re = Poles.at(k-1).real();
|
|
double im = Poles.at(k-1).imag();
|
|
double B = -2.0*re;
|
|
double C = re*re + im*im;
|
|
|
|
double H = C + 4.0*Q*Q;
|
|
double E = (1.0/B)*sqrt(0.5*(H+sqrt(H*H-(4.0*B*B*Q*Q))));
|
|
double F = (B*E)/Q;
|
|
double D = 0.5*(F+sqrt(F*F-4.0));
|
|
|
|
qDebug()<<D<<E<<Q;
|
|
|
|
double rho = Kv1*sqrt(C)/Q;
|
|
double beta = D/E;
|
|
double gamma = D*D;
|
|
|
|
C1 = 10.0/F0;
|
|
R1 = 2.0/(rho*W0*C1);
|
|
R2 = 2.0/((-beta+sqrt((rho-beta)*(rho-beta)+8.0*gamma))*W0*C1);
|
|
R3 = (1.0/R1+1.0/R2)/(gamma*W0*W0*C1*C1);
|
|
R4 = 2.0*R3;
|
|
|
|
RC_elements current_section;
|
|
current_section.N = cnt;
|
|
current_section.R1 = 1000*R1;
|
|
current_section.R2 = 1000*R2;
|
|
current_section.R3 = 1000*R3;
|
|
current_section.R4 = 1000*R4;
|
|
current_section.R5 = 0;
|
|
current_section.R6 = 0;
|
|
current_section.C1 = C1;
|
|
current_section.C2 = C1;
|
|
Sections.append(current_section);
|
|
|
|
cnt++;
|
|
|
|
beta = 1.0/(D*E);
|
|
gamma = 1.0/(D*D);
|
|
C1 = 10.0/F0;
|
|
R1 = 2.0/(rho*W0*C1);
|
|
R2 = 2.0/((-beta+sqrt((rho-beta)*(rho-beta)+8.0*gamma))*W0*C1);
|
|
R3 = (1.0/R1+1.0/R2)/(gamma*W0*W0*C1*C1);
|
|
R4 = 2.0*R3;
|
|
|
|
current_section.N = cnt;
|
|
current_section.R1 = 1000*R1;
|
|
current_section.R2 = 1000*R2;
|
|
current_section.R3 = 1000*R3;
|
|
current_section.R4 = 1000*R4;
|
|
current_section.R5 = 0;
|
|
current_section.R6 = 0;
|
|
current_section.C1 = C1;
|
|
current_section.C2 = C1;
|
|
Sections.append(current_section);
|
|
|
|
cnt++;
|
|
}
|
|
|
|
|
|
if (order%2 != 0) { // Need to implement first-order section
|
|
|
|
double R1,R2,R3,R4,C1;
|
|
|
|
int k = order/2 + 1;
|
|
double re = Poles.at(k-1).real();
|
|
double C = -re;
|
|
double rho = Kv1*C/Q;
|
|
double beta = C/Q;
|
|
double gamma = 1.0;
|
|
|
|
C1 = 10.0/F0;
|
|
R1 = 2.0/(rho*W0*C1);
|
|
R2 = 2.0/((-beta+sqrt((rho-beta)*(rho-beta)+8.0*gamma))*W0*C1);
|
|
R3 = (1.0/R1+1.0/R2)/(gamma*W0*W0*C1*C1);
|
|
R4 = 2.0*R3;
|
|
|
|
RC_elements current_section;
|
|
current_section.N = cnt;
|
|
current_section.R1 = 1000*R1;
|
|
current_section.R2 = 1000*R2;
|
|
current_section.R3 = 1000*R3;
|
|
current_section.R4 = 1000*R4;
|
|
current_section.R5 = 0;
|
|
current_section.R6 = 0;
|
|
current_section.C1 = C1;
|
|
current_section.C2 = C1;
|
|
Sections.append(current_section);
|
|
}
|
|
|
|
}
|
|
|
|
void SallenKey::calcBandStop()
|
|
{
|
|
|
|
}
|
|
|
|
void SallenKey::createHighPassSchematic(QString &s)
|
|
{
|
|
int const N_R=4; // number of resisitors in 2-order Sallen-Key stage
|
|
int const N_C=2; // number of capacitors in 2-order Sallen-Key stage
|
|
RC_elements stage;
|
|
int dx = 0;
|
|
int N2ord = order/2; // number of 2-nd order stages
|
|
int N1stOrd = order%2; // number of 1-st order stages
|
|
|
|
s += "<Qucs Schematic ";
|
|
s += PACKAGE_VERSION;
|
|
s += ">\n";
|
|
s += "<Components>\n";
|
|
s += QStringLiteral("<.AC AC1 1 30 410 0 61 0 0 \"lin\" 1 \"1 Hz\" 1 \"%1 kHz\" 1 \"501\" 1 \"no\" 0>\n").arg((10.0*Fc)/1000.0);
|
|
s += "<.DC DC1 1 280 410 0 61 0 0 \"26.85\" 0 \"0.001\" 0 \"1 pA\" 0 \"1 uV\" 0 \"no\" 0 \"150\" 0 \"no\" 0 \"none\" 0 \"CroutLU\" 0>\n";
|
|
s += "<Eqn Eqn1 1 270 540 -30 14 0 0 \"K=(out.v/in.v)\" 1 \"yes\" 0>\n";
|
|
s += QStringLiteral("<Vac V1 1 %1 260 18 -26 0 1 \"1 V\" 1 \"1 kHz\" 0 \"0\" 0 \"0\" 0>\n").arg(20+dx);
|
|
s += QStringLiteral("<GND * 1 %1 290 0 0 0 0>\n").arg(20+dx);
|
|
for (int i=1; i<=N2ord; i++) {
|
|
stage = Sections.at(i-1);
|
|
QString suffix1, suffix2;
|
|
double C1 = autoscaleCapacitor(stage.C1,suffix1);
|
|
double C2 = autoscaleCapacitor(stage.C2,suffix2);
|
|
s += QStringLiteral("<OpAmp OP%1 1 %2 160 -26 42 0 0 \"1e6\" 1 \"15 V\" 0>\n").arg(stage.N).arg(370+dx);
|
|
s += QStringLiteral("<GND * 1 %1 270 0 0 0 0>\n").arg(270+dx);
|
|
s += QStringLiteral("<GND * 1 %1 370 0 0 0 0>\n").arg(320+dx);
|
|
s += QStringLiteral("<R R%1 1 %2 340 15 -26 0 1 \"%3k\" 1 \"26.85\" 0 \"0.0\" 0 \"0.0\" 0 \"26.85\" 0 \"european\" 0>\n").arg(3+(i-1)*N_R).arg(320+dx).arg(stage.R3,0,'f',3);
|
|
s += QStringLiteral("<C C%1 1 %2 190 -26 -45 1 0 \"%3%4\" 1 \"\" 0 \"neutral\" 0>\n").arg(2+(i-1)*N_C).arg(200+dx).arg(C2,0,'f',3).arg(suffix2);
|
|
s += QStringLiteral("<C C%1 1 %2 190 -26 17 0 0 \"%3%4\" 1 \"\" 0 \"neutral\" 0>\n").arg(1+(i-1)*N_C).arg(100+dx).arg(C1,0,'f',3).arg(suffix1);
|
|
s += QStringLiteral("<R R%1 1 %2 240 -75 -26 1 1 \"%3k\" 1 \"26.85\" 0 \"0.0\" 0 \"0.0\" 0 \"26.85\" 0 \"european\" 0>\n").arg(2+(i-1)*N_R).arg(270+dx).arg(stage.R2,0,'f',3);
|
|
s += QStringLiteral("<R R%1 1 %2 70 -26 15 0 0 \"%3k\" 1 \"26.85\" 0 \"0.0\" 0 \"0.0\" 0 \"26.85\" 0 \"european\" 0>\n").arg(1+(i-1)*N_R).arg(330+dx).arg(stage.R1,0,'f',3);
|
|
s += QStringLiteral("<R R%1 1 %2 260 -26 15 1 2 \"%3k\" 1 \"26.85\" 0 \"0.0\" 0 \"0.0\" 0 \"26.85\" 0 \"european\" 0>\n").arg(4+(i-1)*N_R).arg(410+dx).arg(stage.R4,0,'f',3);
|
|
dx += 510;
|
|
}
|
|
|
|
if (N1stOrd!=0) {
|
|
createFirstOrderComponentsHPF(s,Sections.last(),dx);
|
|
}
|
|
|
|
s += "</Components>\n";
|
|
s += "<Wires>\n";
|
|
dx = 0;
|
|
s += QStringLiteral("<%1 190 %2 230 \"\" 0 0 0 \"\">\n").arg(20+dx).arg(20+dx);
|
|
s += QStringLiteral("<%1 190 %2 190 \"in\" %3 160 18 \"\">\n").arg(20+dx).arg(70+dx).arg(70+dx);
|
|
for (int i=1; i<=N2ord; i++) {
|
|
if (i!=1) {
|
|
s += QStringLiteral("<%1 190 %2 190 \"\" 0 0 0 \"\">\n").arg(dx-20).arg(70+dx);
|
|
s += QStringLiteral("<%1 190 %2 160 \"\" 0 0 0 \"\">\n").arg(dx-20).arg(dx-20);
|
|
s += QStringLiteral("<%1 160 %2 160 \"\" 0 0 0 \"\">\n").arg(dx-20).arg(dx-50);
|
|
}
|
|
s += QStringLiteral("<%1 70 %2 70 \"\" 0 0 0 \"\">\n").arg(360+dx).arg(460+dx);
|
|
if ((2*i)==order) {
|
|
s += QStringLiteral("<%1 70 %2 160 \"out\" %3 90 51 \"\">\n").arg(460+dx).arg(460+dx).arg(490+dx);
|
|
} else {
|
|
s += QStringLiteral("<%1 70 %2 160 \"\" 0 0 0 \"\">\n").arg(460+dx).arg(460+dx);
|
|
}
|
|
s += QStringLiteral("<%1 160 %2 160 \"\" 0 0 0 \"\">\n").arg(410+dx).arg(460+dx);
|
|
s += QStringLiteral("<%1 260 %2 260 \"\" 0 0 0 \"\">\n").arg(440+dx).arg(460+dx);
|
|
s += QStringLiteral("<%1 160 %2 260 \"\" 0 0 0 \"\">\n").arg(460+dx).arg(460+dx);
|
|
s += QStringLiteral("<%1 190 %2 210 \"\" 0 0 0 \"\">\n").arg(270+dx).arg(270+dx);
|
|
s += QStringLiteral("<%1 190 %2 190 \"\" 0 0 0 \"\">\n").arg(230+dx).arg(270+dx);
|
|
s += QStringLiteral("<%1 190 %2 190 \"\" 0 0 0 \"\">\n").arg(130+dx).arg(150+dx);
|
|
s += QStringLiteral("<%1 190 %2 190 \"\" 0 0 0 \"\">\n").arg(150+dx).arg(170+dx);
|
|
s += QStringLiteral("<%1 70 %2 190 \"\" 0 0 0 \"\">\n").arg(150+dx).arg(150+dx);
|
|
s += QStringLiteral("<%1 70 %2 70 \"\" 0 0 0 \"\">\n").arg(150+dx).arg(300+dx);
|
|
s += QStringLiteral("<%1 140 %2 190 \"\" 0 0 0 \"\">\n").arg(270+dx).arg(270+dx);
|
|
s += QStringLiteral("<%1 140 %2 140 \"\" 0 0 0 \"\">\n").arg(270+dx).arg(340+dx);
|
|
s += QStringLiteral("<%1 180 %2 260 \"\" 0 0 0 \"\">\n").arg(320+dx).arg(320+dx);
|
|
s += QStringLiteral("<%1 180 %2 180 \"\" 0 0 0 \"\">\n").arg(320+dx).arg(340+dx);
|
|
s += QStringLiteral("<%1 260 %2 260 \"\" 0 0 0 \"\">\n").arg(320+dx).arg(380+dx);
|
|
s += QStringLiteral("<%1 260 %2 310 \"\" 0 0 0 \"\">\n").arg(320+dx).arg(320+dx);
|
|
dx += 510;
|
|
}
|
|
|
|
if (N1stOrd!=0) {
|
|
createFirstOrderWires(s,dx,160);
|
|
}
|
|
|
|
s += "</Wires>\n";
|
|
}
|
|
|
|
void SallenKey::createLowPassSchematic(QString &s)
|
|
{
|
|
int const N_R=4; // number of resisitors in 2-order Sallen-Key stage
|
|
int const N_C=2; // number of capacitors in 2-order Sallen-Key stage
|
|
RC_elements stage;
|
|
int dx = 0;
|
|
int N2ord = order/2; // number of 2-nd order stages
|
|
int N1stOrd = order%2; // number of 1-st order stages
|
|
|
|
s += "<Qucs Schematic ";
|
|
s += PACKAGE_VERSION;
|
|
s += ">\n";
|
|
s += "<Components>\n";
|
|
s += QStringLiteral("<.AC AC1 1 30 410 0 61 0 0 \"lin\" 1 \"1 Hz\" 1 \"%1 kHz\" 1 \"501\" 1 \"no\" 0>\n").arg((10.0*Fc)/1000.0);
|
|
s += "<.DC DC1 1 280 410 0 61 0 0 \"26.85\" 0 \"0.001\" 0 \"1 pA\" 0 \"1 uV\" 0 \"no\" 0 \"150\" 0 \"no\" 0 \"none\" 0 \"CroutLU\" 0>\n";
|
|
s += "<Eqn Eqn1 1 270 540 -30 14 0 0 \"K=(out.v/in.v)\" 1 \"yes\" 0>\n";
|
|
s += QStringLiteral("<Vac V1 1 %1 260 18 -26 0 1 \"1 V\" 1 \"1 kHz\" 0 \"0\" 0 \"0\" 0>\n").arg(20+dx);
|
|
s += QStringLiteral("<GND * 1 %1 290 0 0 0 0>\n").arg(20+dx);
|
|
for (int i=1; i<=N2ord; i++) {
|
|
stage = Sections.at(i-1);
|
|
//qDebug()<<stage.N;
|
|
QString suffix1, suffix2;
|
|
double C1 = autoscaleCapacitor(stage.C1,suffix1);
|
|
double C2 = autoscaleCapacitor(stage.C2,suffix2);
|
|
s += QStringLiteral("<OpAmp OP%1 1 %2 160 -26 42 0 0 \"1e6\" 1 \"15 V\" 0>\n").arg(stage.N).arg(370+dx);
|
|
s += QStringLiteral("<GND * 1 %1 270 0 0 0 0>\n").arg(270+dx);
|
|
s += QStringLiteral("<GND * 1 %1 370 0 0 0 0>\n").arg(320+dx);
|
|
s += QStringLiteral("<C C%1 1 %2 70 -26 15 0 0 \"%3%4\" 1 \"\" 0 \"neutral\" 0>\n").arg(2+(i-1)*N_C).arg(330+dx).arg(C2,0,'f',3).arg(suffix2);
|
|
s += QStringLiteral("<C C%1 1 %2 240 -75 -26 1 1 \"%3%4\" 1 \"\" 0 \"neutral\" 0>\n").arg(1+(i-1)*N_C).arg(270+dx).arg(C1,0,'f',3).arg(suffix1);
|
|
s += QStringLiteral("<R R%1 1 %2 190 -26 -45 1 0 \"%3k\" 1 \"26.85\" 0 \"0.0\" 0 \"0.0\" 0 \"26.85\" 0 \"european\" 0>\n").arg(2+(i-1)*N_R).arg(200+dx).arg(stage.R2,0,'f',3);
|
|
s += QStringLiteral("<R R%1 1 %2 190 -26 17 0 0 \"%3k\" 1 \"26.85\" 0 \"0.0\" 0 \"0.0\" 0 \"26.85\" 0 \"european\" 0>\n").arg(1+(i-1)*N_R).arg(100+dx).arg(stage.R1,0,'f',3);
|
|
s += QStringLiteral("<R R%1 1 %2 340 15 -26 0 1 \"%3k\" 1 \"26.85\" 0 \"0.0\" 0 \"0.0\" 0 \"26.85\" 0 \"european\" 0>\n").arg(3+(i-1)*N_R).arg(320+dx).arg(stage.R3,0,'f',3);
|
|
s += QStringLiteral("<R R%1 1 %2 260 -26 15 1 2 \"%3k\" 1 \"26.85\" 0 \"0.0\" 0 \"0.0\" 0 \"26.85\" 0 \"european\" 0>\n").arg(4+(i-1)*N_R).arg(410+dx).arg(stage.R4,0,'f',3);
|
|
dx += 510;
|
|
}
|
|
|
|
if (N1stOrd!=0) {
|
|
createFirstOrderComponentsLPF(s,Sections.last(),dx);
|
|
}
|
|
|
|
s += "</Components>\n";
|
|
s += "<Wires>\n";
|
|
dx = 0;
|
|
s += QStringLiteral("<%1 190 %2 230 \"\" 0 0 0 \"\">\n").arg(20+dx).arg(20+dx);
|
|
s += QStringLiteral("<%1 190 %2 190 \"in\" %3 160 18 \"\">\n").arg(20+dx).arg(70+dx).arg(70+dx);
|
|
for (int i=1; i<=N2ord; i++) {
|
|
if (i!=1) {
|
|
s += QStringLiteral("<%1 190 %2 190 \"\" 0 0 0 \"\">\n").arg(dx-20).arg(70+dx);
|
|
s += QStringLiteral("<%1 190 %2 160 \"\" 0 0 0 \"\">\n").arg(dx-20).arg(dx-20);
|
|
s += QStringLiteral("<%1 160 %2 160 \"\" 0 0 0 \"\">\n").arg(dx-20).arg(dx-50);
|
|
}
|
|
s += QStringLiteral("<%1 70 %2 70 \"\" 0 0 0 \"\">\n").arg(360+dx).arg(460+dx);
|
|
if ((2*i)==order) {
|
|
s += QStringLiteral("<%1 70 %2 160 \"out\" %3 90 51 \"\">\n").arg(460+dx).arg(460+dx).arg(490+dx);
|
|
} else {
|
|
s += QStringLiteral("<%1 70 %2 160 \"\" 0 0 0 \"\">\n").arg(460+dx).arg(460+dx);
|
|
}
|
|
s += QStringLiteral("<%1 160 %2 160 \"\" 0 0 0 \"\">\n").arg(410+dx).arg(460+dx);
|
|
s += QStringLiteral("<%1 260 %2 260 \"\" 0 0 0 \"\">\n").arg(440+dx).arg(460+dx);
|
|
s += QStringLiteral("<%1 160 %2 260 \"\" 0 0 0 \"\">\n").arg(460+dx).arg(460+dx);
|
|
s += QStringLiteral("<%1 190 %2 210 \"\" 0 0 0 \"\">\n").arg(270+dx).arg(270+dx);
|
|
s += QStringLiteral("<%1 190 %2 190 \"\" 0 0 0 \"\">\n").arg(230+dx).arg(270+dx);
|
|
s += QStringLiteral("<%1 190 %2 190 \"\" 0 0 0 \"\">\n").arg(130+dx).arg(150+dx);
|
|
s += QStringLiteral("<%1 190 %2 190 \"\" 0 0 0 \"\">\n").arg(150+dx).arg(170+dx);
|
|
s += QStringLiteral("<%1 70 %2 190 \"\" 0 0 0 \"\">\n").arg(150+dx).arg(150+dx);
|
|
s += QStringLiteral("<%1 70 %2 70 \"\" 0 0 0 \"\">\n").arg(150+dx).arg(300+dx);
|
|
s += QStringLiteral("<%1 140 %2 190 \"\" 0 0 0 \"\">\n").arg(270+dx).arg(270+dx);
|
|
s += QStringLiteral("<%1 140 %2 140 \"\" 0 0 0 \"\">\n").arg(270+dx).arg(340+dx);
|
|
s += QStringLiteral("<%1 180 %2 260 \"\" 0 0 0 \"\">\n").arg(320+dx).arg(320+dx);
|
|
s += QStringLiteral("<%1 180 %2 180 \"\" 0 0 0 \"\">\n").arg(320+dx).arg(340+dx);
|
|
s += QStringLiteral("<%1 260 %2 260 \"\" 0 0 0 \"\">\n").arg(320+dx).arg(380+dx);
|
|
s += QStringLiteral("<%1 260 %2 310 \"\" 0 0 0 \"\">\n").arg(320+dx).arg(320+dx);
|
|
dx += 510;
|
|
}
|
|
|
|
if (N1stOrd!=0) {
|
|
createFirstOrderWires(s,dx,160);
|
|
}
|
|
|
|
s += "</Wires>\n";
|
|
}
|
|
|
|
void SallenKey::createBandPassSchematic(QString &s)
|
|
{
|
|
RC_elements stage;
|
|
int dx = 0;
|
|
|
|
s += "<Qucs Schematic ";
|
|
s += PACKAGE_VERSION;
|
|
s += ">\n";
|
|
s += "<Components>\n";
|
|
s += QStringLiteral("<.AC AC1 1 30 410 0 61 0 0 \"lin\" 1 \"1 Hz\" 1 \"%1 kHz\" 1 \"501\" 1 \"no\" 0>\n").arg((Fu+1000)/1000.0);
|
|
s += "<.DC DC1 1 280 410 0 61 0 0 \"26.85\" 0 \"0.001\" 0 \"1 pA\" 0 \"1 uV\" 0 \"no\" 0 \"150\" 0 \"no\" 0 \"none\" 0 \"CroutLU\" 0>\n";
|
|
s += "<Eqn Eqn1 1 270 540 -30 14 0 0 \"K=(out.v/in.v)\" 1 \"yes\" 0>\n";
|
|
s += QStringLiteral("<Vac V1 1 %1 260 18 -26 0 1 \"1 V\" 1 \"1 kHz\" 0 \"0\" 0 \"0\" 0>\n").arg(20+dx);
|
|
s += QStringLiteral("<GND * 1 %1 290 0 0 0 0>\n").arg(20+dx);
|
|
for (int i=1; i<=Sections.count(); i++) {
|
|
stage = Sections.at(i-1);
|
|
//qDebug()<<stage.N;
|
|
QString suffix1, suffix2;
|
|
double C1 = autoscaleCapacitor(stage.C1,suffix1);
|
|
double C2 = autoscaleCapacitor(stage.C2,suffix2);
|
|
s += QStringLiteral("<OpAmp OP%1 1 %2 160 -26 42 0 0 \"1e6\" 1 \"15 V\" 0>\n").arg(stage.N).arg(370+dx);
|
|
s += QStringLiteral("<GND * 1 %1 270 0 0 0 0>\n").arg(270+dx);
|
|
s += QStringLiteral("<GND * 1 %1 370 0 0 0 0>\n").arg(320+dx);
|
|
s += QStringLiteral("<GND * 1 %1 310 0 0 0 0>\n").arg(150+dx);
|
|
s += QStringLiteral("<C C%1 1 %2 280 17 -26 0 1 \"%3%4\" 1 \"\" 0 \"neutral\" 0>\n").arg(1+(i-1)*Nc1).arg(150+dx).arg(C1,0,'f',3).arg(suffix1);
|
|
s += QStringLiteral("<C C%1 1 %2 190 -26 -45 1 0 \"%3%4\" 1 \"\" 0 \"neutral\" 0>\n").arg(2+(i-1)*Nr1).arg(200+dx).arg(C2,0,'f',3).arg(suffix2);
|
|
s += QStringLiteral("<R R%1 1 %2 190 -26 17 0 0 \"%3k\" 1 \"26.85\" 0 \"0.0\" 0 \"0.0\" 0 \"26.85\" 0 \"european\" 0>\n").arg(1+(i-1)*Nr1).arg(100+dx).arg(stage.R1,0,'f',3);
|
|
s += QStringLiteral("<R R%1 1 %2 70 -26 15 0 0 \"%3k\" 1 \"26.85\" 0 \"0.0\" 0 \"0.0\" 0 \"26.85\" 0 \"european\" 0>\n").arg(2+(i-1)*Nr1).arg(330+dx).arg(stage.R2,0,'f',3);
|
|
s += QStringLiteral("<R R%1 1 %2 240 -75 -26 1 1 \"%3k\" 1 \"26.85\" 0 \"0.0\" 0 \"0.0\" 0 \"26.85\" 0 \"european\" 0>\n").arg(3+(i-1)*Nr1).arg(270+dx).arg(stage.R3,0,'f',3);
|
|
s += QStringLiteral("<R R%1 1 %2 340 15 -26 0 1 \"%3k\" 1 \"26.85\" 0 \"0.0\" 0 \"0.0\" 0 \"26.85\" 0 \"european\" 0>\n").arg(4+(i-1)*Nr1).arg(320+dx).arg(stage.R4,0,'f',3);
|
|
s += QStringLiteral("<R R%1 1 %2 260 -26 15 1 2 \"%3k\" 1 \"26.85\" 0 \"0.0\" 0 \"0.0\" 0 \"26.85\" 0 \"european\" 0>\n").arg(5+(i-1)*Nr1).arg(410+dx).arg(stage.R4,0,'f',3);
|
|
dx += 510;
|
|
}
|
|
|
|
|
|
s += "</Components>\n";
|
|
s += "<Wires>\n";
|
|
dx = 0;
|
|
s += QStringLiteral("<%1 190 %2 230 \"\" 0 0 0 \"\">\n").arg(20+dx).arg(20+dx);
|
|
s += QStringLiteral("<%1 190 %2 190 \"in\" %3 160 18 \"\">\n").arg(20+dx).arg(70+dx).arg(70+dx);
|
|
for (int i=1; i<=Sections.count(); i++) {
|
|
if (i!=1) {
|
|
s += QStringLiteral("<%1 190 %2 190 \"\" 0 0 0 \"\">\n").arg(dx-20).arg(70+dx);
|
|
s += QStringLiteral("<%1 190 %2 160 \"\" 0 0 0 \"\">\n").arg(dx-20).arg(dx-20);
|
|
s += QStringLiteral("<%1 160 %2 160 \"\" 0 0 0 \"\">\n").arg(dx-20).arg(dx-50);
|
|
}
|
|
s += QStringLiteral("<%1 70 %2 70 \"\" 0 0 0 \"\">\n").arg(360+dx).arg(460+dx);
|
|
if (i==order) {
|
|
s += QStringLiteral("<%1 70 %2 160 \"out\" %3 90 51 \"\">\n").arg(460+dx).arg(460+dx).arg(490+dx);
|
|
} else {
|
|
s += QStringLiteral("<%1 70 %2 160 \"\" 0 0 0 \"\">\n").arg(460+dx).arg(460+dx);
|
|
}
|
|
s += QStringLiteral("<%1 160 %2 160 \"\" 0 0 0 \"\">\n").arg(410+dx).arg(460+dx);
|
|
s += QStringLiteral("<%1 260 %2 260 \"\" 0 0 0 \"\">\n").arg(440+dx).arg(460+dx);
|
|
s += QStringLiteral("<%1 160 %2 260 \"\" 0 0 0 \"\">\n").arg(460+dx).arg(460+dx);
|
|
s += QStringLiteral("<%1 190 %2 210 \"\" 0 0 0 \"\">\n").arg(270+dx).arg(270+dx);
|
|
s += QStringLiteral("<%1 190 %2 190 \"\" 0 0 0 \"\">\n").arg(230+dx).arg(270+dx);
|
|
s += QStringLiteral("<%1 190 %2 190 \"\" 0 0 0 \"\">\n").arg(130+dx).arg(150+dx);
|
|
s += QStringLiteral("<%1 190 %2 190 \"\" 0 0 0 \"\">\n").arg(150+dx).arg(170+dx);
|
|
s += QStringLiteral("<%1 70 %2 190 \"\" 0 0 0 \"\">\n").arg(150+dx).arg(150+dx);
|
|
s += QStringLiteral("<%1 70 %2 70 \"\" 0 0 0 \"\">\n").arg(150+dx).arg(300+dx);
|
|
s += QStringLiteral("<%1 140 %2 190 \"\" 0 0 0 \"\">\n").arg(270+dx).arg(270+dx);
|
|
s += QStringLiteral("<%1 140 %2 140 \"\" 0 0 0 \"\">\n").arg(270+dx).arg(340+dx);
|
|
s += QStringLiteral("<%1 180 %2 260 \"\" 0 0 0 \"\">\n").arg(320+dx).arg(320+dx);
|
|
s += QStringLiteral("<%1 180 %2 180 \"\" 0 0 0 \"\">\n").arg(320+dx).arg(340+dx);
|
|
s += QStringLiteral("<%1 260 %2 260 \"\" 0 0 0 \"\">\n").arg(320+dx).arg(380+dx);
|
|
s += QStringLiteral("<%1 260 %2 310 \"\" 0 0 0 \"\">\n").arg(320+dx).arg(320+dx);
|
|
s += QStringLiteral("<%1 190 %2 250 \"\" 0 0 0 \"\">\n").arg(150+dx).arg(150+dx);
|
|
dx += 510;
|
|
}
|
|
|
|
|
|
s += "</Wires>\n";
|
|
}
|
|
|
|
void SallenKey::createBandStopSchematic(QString &s)
|
|
{
|
|
Q_UNUSED(s);
|
|
|
|
}
|
|
|
|
|