/* * qucspowercombiningtool.cpp - Power combining tool implementation * * copyright (C) 2017 Andres Martinez-Mera * * This 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, or (at your option) * any later version. * * This software 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 General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this package; see the file COPYING. If not, write to * the Free Software Foundation, Inc., 51 Franklin Street - Fifth Floor, * Boston, MA 02110-1301, USA. * * */ #ifdef HAVE_CONFIG_H #include #endif #include #include "qucspowercombiningtool.h" #include "../qucs/qucs.h" #include "../qucs/misc.h" #include "../qucs-filter/material_props.h" #include "../qucs/extsimkernels/spicecompat.h" //--------------------------------------------------------- // Constructor. It sets up the user interface QucsPowerCombiningTool::QucsPowerCombiningTool() { setWindowTitle("Qucs Power Combining Tool " PACKAGE_VERSION); // set application icon setWindowIcon(QPixmap(":/images/bitmaps/big.qucs.xpm")); centralWidget = new QWidget(); QHBoxLayout *hbox = new QHBoxLayout(); SpecificationsgroupBox = new QGroupBox("Implementation"); MicrostripgroupBox = new QGroupBox("Microstrip substrate"); ImagegroupBox = new QGroupBox("Preview"); //Add elements to SpecificationsgroupBox QVBoxLayout *VboxImplementation = new QVBoxLayout(); //Topology QHBoxLayout *hboxImpl1 = new QHBoxLayout(); TopoLabel = new QLabel("Topology"); TopoCombo = new QComboBox(); TopoCombo->addItem("Wilkinson"); TopoCombo->addItem("Multistage Wilkinson"); TopoCombo->addItem("T-junction"); TopoCombo->addItem("Branchline"); TopoCombo->addItem("Double box branchline"); TopoCombo->addItem("Bagley"); TopoCombo->addItem("Gysel"); TopoCombo->addItem("Travelling Wave"); TopoCombo->addItem("Tree"); hboxImpl1->addWidget(TopoLabel); hboxImpl1->addWidget(TopoCombo); VboxImplementation->addLayout(hboxImpl1); // microstrip circuits cannot be implemented with SPICE if (QucsSettings.DefaultSimulator != spicecompat::simQucsator) { QStandardItemModel *model = qobject_cast(TopoCombo->model()); Q_ASSERT(model != nullptr); for (int i = 2; i < TopoCombo->count(); i++) { QStandardItem *itm = model->item(i); itm->setFlags(itm->flags() & ~Qt::ItemIsEnabled); } } //Number of outputs QHBoxLayout *hboxImpl2 = new QHBoxLayout(); NLabel =new QLabel("Number of outputs"); BranchesCombo = new QComboBox(); BranchesCombo->addItem("2"); BranchesCombo->setEnabled(false); BranchesCombo->setFixedWidth(60); hboxImpl2->addWidget(NLabel); hboxImpl2->addWidget(BranchesCombo); VboxImplementation->addLayout(hboxImpl2); //Ref impedance QHBoxLayout *hboxImpl3 = new QHBoxLayout(); RefImp = new QLabel("Z0"); RefImplineEdit = new QLineEdit("50"); RefImplineEdit->setFixedWidth(40); OhmLabel = new QLabel(QChar(0xa9, 0x03)); OhmLabel->setFixedWidth(15); hboxImpl3->addWidget(RefImp); hboxImpl3->addWidget(RefImplineEdit); hboxImpl3->addWidget(OhmLabel); VboxImplementation->addLayout(hboxImpl3); //Frequency QHBoxLayout *hboxImpl4 = new QHBoxLayout(); FreqLabel = new QLabel("Frequency"); FreqlineEdit = new QLineEdit("1"); FreqlineEdit->setFixedWidth(40); FreqScaleCombo=new QComboBox(); FreqScaleCombo->addItem("GHz"); FreqScaleCombo->addItem("MHz"); FreqScaleCombo->addItem("kHz"); FreqScaleCombo->addItem("Hz"); FreqScaleCombo->setFixedWidth(60); hboxImpl4->addWidget(FreqLabel); hboxImpl4->addWidget(FreqlineEdit); hboxImpl4->addWidget(FreqScaleCombo); VboxImplementation->addLayout(hboxImpl4); // Output power ratio QHBoxLayout *hboxImpl5 = new QHBoxLayout(); K1Label = new QLabel("Output Power ratio"); K1lineEdit=new QLineEdit("0"); K1lineEdit->setFixedWidth(40); K1LabeldB = new QLabel("dB"); K1LabeldB->setFixedWidth(20); hboxImpl5->addWidget(K1Label); hboxImpl5->addWidget(K1lineEdit); hboxImpl5->addWidget(K1LabeldB); VboxImplementation->addLayout(hboxImpl5); // Number of stages QHBoxLayout *hboxImpl5_ = new QHBoxLayout(); NStagesCombo = new QComboBox(); NStagesCombo->addItem("2"); NStagesCombo->setFixedWidth(60); NStagesLabel = new QLabel("Number of stages"); hboxImpl5_->addWidget(NStagesLabel); hboxImpl5_->addWidget(NStagesCombo); VboxImplementation->addLayout(hboxImpl5_); NStagesLabel->hide(); NStagesCombo->hide(); //Ideal transmission line attenuation coeffient QHBoxLayout *hboxImplAlpha = new QHBoxLayout(); AlphaLabel = new QLabel("Attenuation coefficient"); AlphalineEdit=new QLineEdit("0"); AlphadBLabel = new QLabel("dB/m"); hboxImplAlpha->addWidget(AlphaLabel); hboxImplAlpha->addWidget(AlphalineEdit); hboxImplAlpha->addWidget(AlphadBLabel); AlphadBLabel->setVisible(false); AlphaLabel->setVisible(false); AlphalineEdit->setVisible(false); VboxImplementation->addLayout(hboxImplAlpha); QHBoxLayout *hboxLengthUnit = new QHBoxLayout(); UnitsLabel = new QLabel("Length unit"); UnitsCombo=new QComboBox(); UnitsCombo->addItem("mm"); UnitsCombo->addItem("mil"); UnitsCombo->addItem("um"); UnitsCombo->addItem("nm"); UnitsCombo->addItem("inch"); UnitsCombo->addItem("ft"); UnitsCombo->addItem("m"); UnitsCombo->setFixedWidth(60); hboxLengthUnit->addWidget(UnitsLabel); hboxLengthUnit->addWidget(UnitsCombo); VboxImplementation->addLayout(hboxLengthUnit); //Add S-param block? QHBoxLayout *hboxImpl6 = new QHBoxLayout(); AddSparcheckBox=new QCheckBox("Add S-parameter simulation"); AddSparcheckBox->setChecked(true); hboxImpl6->addWidget(AddSparcheckBox); VboxImplementation->addLayout(hboxImpl6); //Implementation. Mutually exclusive radiobuttons for selecting ideal transmission lines, microstrip tech, o CLC pi aproximations of quarter-wave lines ImplementationgroupBox = new QGroupBox("Implementation"); QHBoxLayout *hboxImpl7 = new QHBoxLayout();//Layout for the groupbox QVBoxLayout *vboxImpl = new QVBoxLayout();//Layout for allocating the radiobuttons IdealTLradioButton=new QRadioButton("Ideal TL");//Synthesize Microstrip MicrostripradioButton=new QRadioButton("Microstrip");//Synthesize Microstrip LumpedElementsradioButton=new QRadioButton("Lumped elements");//Lumped elements implementation //Add widgets to the layout vboxImpl->addWidget(IdealTLradioButton); vboxImpl->addWidget(MicrostripradioButton); vboxImpl->addWidget(LumpedElementsradioButton); IdealTLradioButton->setChecked(true);//Activate ideal TL by default ImplementationgroupBox->setLayout(vboxImpl);//Add layout to the groupbox hboxImpl7->addWidget(ImplementationgroupBox); VboxImplementation->addLayout(hboxImpl7); //Go! Button QHBoxLayout *hboxImpl8 = new QHBoxLayout(); GenerateButton = new QPushButton("Generate schematic"); hboxImpl8->addWidget(GenerateButton); VboxImplementation->addLayout(hboxImpl8); // Add elements to MicrostripgroupBox QVBoxLayout *VboxMicrostrip = new QVBoxLayout(); //Relative permitivity QHBoxLayout *hboxMicr1 = new QHBoxLayout(); RelPermlabel = new QLabel("Relative permitivity"); RelPermcomboBox = new QComboBox(); RelPermcomboBox->setEditable(true); const char **p = List_er; while(*(++p)) RelPermcomboBox->addItem(*p); hboxMicr1->addWidget(RelPermlabel); hboxMicr1->addWidget(RelPermcomboBox); VboxMicrostrip->addLayout(hboxMicr1); //Substrate height QHBoxLayout *hboxMicr2 = new QHBoxLayout(); SubstrateHeightlabel = new QLabel("Substrate heigth"); SubstrateHeightlineEdit = new QLineEdit("1.0"); SubstrateMMlabel = new QLabel("mm"); hboxMicr2->addWidget(SubstrateHeightlabel); hboxMicr2->addWidget(SubstrateHeightlineEdit); hboxMicr2->addWidget(SubstrateMMlabel); VboxMicrostrip->addLayout(hboxMicr2); //Metal thickness QHBoxLayout *hboxMicr3 = new QHBoxLayout(); ThicknessLabel = new QLabel("Metal thickness"); ThicknesslineEdit = new QLineEdit("12.5"); ThicknessumLabel = new QLabel("um"); hboxMicr3->addWidget(ThicknessLabel); hboxMicr3->addWidget(ThicknesslineEdit); hboxMicr3->addWidget(ThicknessumLabel); VboxMicrostrip->addLayout(hboxMicr3); //Minimum width QHBoxLayout *hboxMicr4 = new QHBoxLayout(); MinWidthLabel= new QLabel("Minimum width"); MinWidthlineEdit = new QLineEdit("0.4"); MinWidthmmLabel = new QLabel("mm"); hboxMicr4->addWidget(MinWidthLabel); hboxMicr4->addWidget(MinWidthlineEdit); hboxMicr4->addWidget(MinWidthmmLabel); VboxMicrostrip->addLayout(hboxMicr4); //Maximum width QHBoxLayout *hboxMicr5 = new QHBoxLayout(); MaxWidthLabel= new QLabel("Maximum width"); MaxWidthlineEdit = new QLineEdit("5"); MaxWidthmmLabel = new QLabel("mm"); hboxMicr5->addWidget(MaxWidthLabel); hboxMicr5->addWidget(MaxWidthlineEdit); hboxMicr5->addWidget(MaxWidthmmLabel); VboxMicrostrip->addLayout(hboxMicr5); //tan(delta) QHBoxLayout *hboxMicr6 = new QHBoxLayout(); tanDLabel = new QLabel("tanD"); tanDlineEdit = new QLineEdit("0.0125"); hboxMicr6->addWidget(tanDLabel); hboxMicr6->addWidget(tanDlineEdit); VboxMicrostrip->addLayout(hboxMicr6); //Resistivity QHBoxLayout *hboxMicr7 = new QHBoxLayout(); ResistivityLabel = new QLabel("Resistivity"); ResistivitylineEdit = new QLineEdit("2.43902e-08"); hboxMicr7->addWidget(ResistivityLabel); hboxMicr7->addWidget(ResistivitylineEdit); VboxMicrostrip->addLayout(hboxMicr7); //Roughness QHBoxLayout *hboxMicr8 = new QHBoxLayout(); RoughnessLabel = new QLabel("Roughness"); RoughnesslineEdit = new QLineEdit("0.15e-6"); hboxMicr8->addWidget(RoughnessLabel); hboxMicr8->addWidget(RoughnesslineEdit); VboxMicrostrip->addLayout(hboxMicr8); QSize sz; QString s1 = ":/bitmaps/Wilkinson_idealTL.svg"; QGridLayout * imgLayout = new QGridLayout(); imgWidget = new QSvgWidget(s1); imgWidget->setStyleSheet("background-color: white;"); sz = imgWidget->size(); imgWidget->setFixedSize(.6*sz); imgLayout->addWidget(imgWidget); ImagegroupBox->setLayout(imgLayout); imgLayout->setAlignment(imgWidget, Qt::AlignHCenter); SpecificationsgroupBox->setLayout(VboxImplementation); MicrostripgroupBox->setLayout(VboxMicrostrip); hbox->addWidget(SpecificationsgroupBox); hbox->addWidget(ImagegroupBox); hbox->addWidget(MicrostripgroupBox); centralWidget->setLayout(hbox); setCentralWidget(centralWidget); MicrostripgroupBox->setEnabled(false); statusBar = new QStatusBar(); setStatusBar(statusBar); connect(GenerateButton, SIGNAL(clicked()), SLOT(on_GenerateButton_clicked())); connect(TopoCombo, SIGNAL(currentIndexChanged(int)), SLOT(on_TopoCombo_currentIndexChanged(int))); connect(MicrostripradioButton, SIGNAL(clicked()), SLOT(on_MicrostripradioButton_clicked())); connect(LumpedElementsradioButton, SIGNAL(clicked()), SLOT(on_LCRadioButton_clicked())); connect(IdealTLradioButton, SIGNAL(clicked()), SLOT(on_IdealTLRadioButton_clicked())); if (QucsSettings.DefaultSimulator != spicecompat::simQucsator) { IdealTLradioButton->setEnabled(false); MicrostripradioButton->setEnabled(false); LumpedElementsradioButton->setChecked(true); on_LCRadioButton_clicked(); } } //------------------------------------------------ // Destructor QucsPowerCombiningTool::~QucsPowerCombiningTool() { } //--------------------------------------------------------------------------- // This function scales the frequency according to 'FreqScaleCombo' combobox double QucsPowerCombiningTool::getScaleFreq() { double exp=1; switch(FreqScaleCombo->currentIndex()) { case 0: exp=9; break; case 1: exp=6; break; case 2: exp=3; break; case 3: exp=1; break; } return pow(10, exp); } //------------------------------------------------------------------ // Update image void QucsPowerCombiningTool::UpdateImage() { int index = TopoCombo->currentIndex(); bool lumpedImplementation = LumpedElementsradioButton->isChecked(); bool microstripImplementation = MicrostripradioButton->isChecked(); //Update image switch (index) { case 0: //Wilkinson if (lumpedImplementation) { imgWidget->load(QStringLiteral(":/bitmaps/WilkinsonLC.svg")); } else { (microstripImplementation) ? imgWidget->load(QStringLiteral(":/bitmaps/Wilkinson_microstrip.svg")) : imgWidget->load(QStringLiteral(":/bitmaps/Wilkinson_idealTL.svg")); } break; case 1: //Mutistage Wilkinson if (lumpedImplementation) { imgWidget->load(QStringLiteral(":/bitmaps/MultistageWilkinsonLC.svg")); } else { (microstripImplementation) ? imgWidget->load(QStringLiteral(":/bitmaps/MultistageWilkinson_microstrip.svg")) : imgWidget->load(QStringLiteral(":/bitmaps/MultistageWilkinson_idealTL.svg")); } break; case 2: //Tee (microstripImplementation) ? imgWidget->load(QStringLiteral(":/bitmaps/Tee_microstrip.svg")) : imgWidget->load(QStringLiteral(":/bitmaps/Tee_idealTL.svg")); break; case 3: //Branch-line (microstripImplementation) ? imgWidget->load(QStringLiteral(":/bitmaps/Branchline_microstrip.svg")) : imgWidget->load(QStringLiteral(":/bitmaps/Branchline_idealTL.svg")); break; case 4: //Double-box branch-line (microstripImplementation) ? imgWidget->load(QStringLiteral(":/bitmaps/DoubleBoxBranchline_microstrip.svg")) : imgWidget->load(QStringLiteral(":/bitmaps/DoubleBoxBranchline_idealTL.svg")); break; case 5: //Bagley power combiner (microstripImplementation) ? imgWidget->load(QStringLiteral(":/bitmaps/Bagley_microstrip.svg")) : imgWidget->load(QStringLiteral(":/bitmaps/Bagley_idealTL.svg")); break; case 6: //Gysel power combiner (microstripImplementation) ? imgWidget->load(QStringLiteral(":/bitmaps/Gysel_microstrip.svg")) : imgWidget->load(QStringLiteral(":/bitmaps/Gysel_idealTL.svg")); break; // ------------- CORPORATE COMBINERS ----------------- case 7: //Travelling wave (microstripImplementation) ? imgWidget->load(QStringLiteral(":/bitmaps/TravellingWave_microstrip.svg")) : imgWidget->load(QStringLiteral(":/bitmaps/TravellingWave_idealTL.svg")); break; case 8: //Tree (microstripImplementation) ? imgWidget->load(QStringLiteral(":/bitmaps/Tree_microstrip.svg")) : imgWidget->load(QStringLiteral(":/bitmaps/Tree_idealTL.svg")); break; } } //------------------------------------------------------------------ // This function changes the window according to the selected topology void QucsPowerCombiningTool::on_TopoCombo_currentIndexChanged(int index) { bool lumpedImplementation = LumpedElementsradioButton->isChecked(); UpdateImage(); // Change settings if ((index==0)|(index==2)||(index==3))//Wilkinson, Tee, Branchline { K1lineEdit->setVisible(true); K1Label->setVisible(true); K1LabeldB->setVisible(true); BranchesCombo->clear(); BranchesCombo->addItem("2"); BranchesCombo->setEnabled(false); BranchesCombo->setEditable(false); } else {//The rest of power combiners do not support unequal power ratio K1lineEdit->setVisible(false); K1Label->setVisible(false); K1LabeldB->setVisible(false); } if ((index == 3)||(index == 4)||(index == 6))//Branchline, double-box branchline and Gysel {//The Gysel power combiner has only two output branches BranchesCombo->clear(); BranchesCombo->addItem("2"); BranchesCombo->setEnabled(false); BranchesCombo->setEditable(false); } if ((index == 0)||(index==1))//Conventional and multistage Wilkinson { LumpedElementsradioButton->setEnabled(true); } else//Only the Wilkinson combiners support the CLC equivalent of a quarter-wavelength line { if (lumpedImplementation) { // Lumped elements checkbox selected, reset to Ideal TL IdealTLradioButton->setChecked(true); } LumpedElementsradioButton->setEnabled(false); } if (index == 1)//Multistage Wilkinson. So far, it is not possible to implement more than 7 stages { NStagesCombo->clear(); NStagesCombo->addItem("2"); NStagesCombo->addItem("3"); NStagesCombo->addItem("4"); NStagesCombo->addItem("5"); NStagesCombo->addItem("6"); NStagesCombo->addItem("7"); NStagesLabel->setVisible(true); NStagesCombo->setVisible(true); BranchesCombo->clear(); BranchesCombo->addItem("2");//2 outputs only BranchesCombo->setEnabled(false); } else//There are no more multistage combiners implemented { NStagesLabel->setVisible(false); NStagesCombo->setVisible(false); } if(index == 5)//Bagley { BranchesCombo->clear(); BranchesCombo->addItem("3"); BranchesCombo->addItem("5"); BranchesCombo->addItem("7"); BranchesCombo->setEditable(true); BranchesCombo->setEnabled(true); } if(index == 7)// Travelling wave { BranchesCombo->clear(); BranchesCombo->addItem("3"); BranchesCombo->addItem("4"); BranchesCombo->addItem("5"); BranchesCombo->addItem("6"); BranchesCombo->setEditable(true);//Let the user to specify an arbitrary number of outputs BranchesCombo->setEnabled(true); } if(index == 8)//Tree { BranchesCombo->clear(); BranchesCombo->addItem("4"); BranchesCombo->addItem("8"); BranchesCombo->addItem("16"); BranchesCombo->setEditable(true);//Let the user to specify an arbitrary number of outputs (power of 2) BranchesCombo->setEnabled(true); } } //--------------------------------------------------------------- // This function reads the input values and calls the different methods for generating a schematic void QucsPowerCombiningTool::on_GenerateButton_clicked() { double err=0; double Z0 = RefImplineEdit->text().toDouble(); int N=BranchesCombo->currentText().toInt(); double K=K1lineEdit->text().toDouble();//Power ratio. Output port 1 K = pow(10, K/20.);//Conversion to natural units double Freq = FreqlineEdit->text().toDouble()*getScaleFreq(); bool SP_block = AddSparcheckBox->isChecked(); int NStages = NStagesCombo->currentText().toInt(); tSubstrate Substrate; double alpha = AlphalineEdit->text().toDouble();//Attenuation coefficient in dB/m bool microcheck = MicrostripradioButton->isChecked(); bool LumpedElements = LumpedElementsradioButton->isChecked();//Lumped element implementation? if (microcheck)//Substrate { Substrate.er=RelPermcomboBox->currentText().section(" ", 0, 0).toDouble(); Substrate.height=SubstrateHeightlineEdit->text().toDouble()*1e-3; Substrate.thickness=ThicknesslineEdit->text().toDouble()*1e-6; Substrate.maxWidth=MaxWidthlineEdit->text().toDouble()*1e-3; Substrate.minWidth=MinWidthlineEdit->text().toDouble()*1e-3; Substrate.resistivity=ResistivitylineEdit->text().toDouble(); Substrate.tand=tanDlineEdit->text().toDouble(); Substrate.roughness=RoughnesslineEdit->text().toDouble(); } switch (TopoCombo->currentIndex())//Topology selection { case 0: // Wilkinson err = Wilkinson(Z0, Freq, K, SP_block, microcheck, Substrate, alpha, LumpedElements); break; case 1: // Multistage Wilkinson err = MultistageWilkinson(Z0, Freq, NStages, SP_block, microcheck, Substrate, alpha, LumpedElements); break; case 2: // Tee err = Tee(Z0, Freq, K*K, SP_block, microcheck, Substrate, alpha); break; case 3: //Branchline err = Branchline(Z0, Freq, K*K, SP_block, microcheck, Substrate, alpha); break; case 4: //Double box branchline err = DoubleBoxBranchline(Z0, Freq, K*K, SP_block, microcheck, Substrate, alpha); break; case 5: //Bagley err = Bagley(Z0, Freq, N, SP_block, microcheck, Substrate, alpha); break; case 6: // Gysel err = Gysel(Z0, Freq, SP_block, microcheck, Substrate, alpha); break; case 7: // Travelling wave err = TravellingWave(Z0, Freq, N, SP_block, microcheck, Substrate, alpha); break; case 8: //Tree err = Tree(Z0, Freq, N, SP_block, microcheck, Substrate, alpha); break; } if(!err)//Checking errors... { statusBar->showMessage(tr("Ready! Use CTRL+V to paste the schematic"), 2000); } else { statusBar->showMessage(tr("Error! The network could not be generated"), 2000); } } //-------------------------------------------------------------------------------- // This function calculates the parameters of the Wilkinson power divider according to the // specifications. It is written outside of Wilkinson() because it is also used by the corporate power // combining functions QString QucsPowerCombiningTool::CalculateWilkinson(double Z0, double K) { // Wilkinson divider design equations double K2 =K*K; double Z3 = Z0*sqrt((K2+1)/(K*K*K)); double Z2 = K2*Z3; double R=Z0*((K2+1)/K); double R2 = Z0*K; double R3 = Z0/K; return QStringLiteral("%1;%2;%3;%4;%5").arg(Z2).arg(Z3).arg(R).arg(R2).arg(R3); } //----------------------------------------------------- // This function calculates a 2Way Wilkinson divider and generates the // schematic int QucsPowerCombiningTool::Wilkinson(double Z0, double Freq, double K, bool SP_block, bool microcheck, tSubstrate Substrate, double Alpha, bool LumpedElements) { double er, width; double lambda4=SPEED_OF_LIGHT/(4*Freq); QString wilkstr = CalculateWilkinson(Z0, K); double Z2 = wilkstr.section(';', 0, 0).toDouble(); double Z3 = wilkstr.section(';', 1, 1).toDouble(); double R = wilkstr.section(';', 2, 2).toDouble(); double R2 = wilkstr.section(';', 3, 3).toDouble(); double R3 = wilkstr.section(';', 4, 4).toDouble(); double C2, C3, CC, C2_, C3_, L2, L3, L2_, L3_, Z4, Z5; if (LumpedElements)//Quarter wave transmission line Pi LC equivalent { double w = 2*M_PI*Freq; L2 = Z2/w; C2 = 1./(L2*w*w); L3 = Z3/w; C3 = 1./(L3*w*w); CC = C2+C3; if (R2 != R3)//Unequal output power rate => requires matching to Z0 { Z4 = Z0*sqrt(K); Z5 = Z0/sqrt(K); L2_ = Z4/w; L3_ = Z5/w; C2_ = 1./(L2_*w*w); C3_ = 1./(L3_*w*w); //Embed the first capacitor of the Pi quarter wave equivalent in the last C of the Wilkinson structure C2 += C2_; C3 += C3_; } } //Qucs schematic QString s = "\n"; s += "\n"; if (SP_block) { //Source s += QStringLiteral("\n").arg(Z0); s += QStringLiteral("\n"); //Output port 1 s += QStringLiteral("\n").arg(Z0); s += QStringLiteral("\n"); //Output port 2 s += QStringLiteral("\n").arg(Z0); s += QStringLiteral("\n"); //S-parameter analysis component QString freq_start = QStringLiteral("%1%2").arg(0.5*FreqlineEdit->text().toDouble()).arg(FreqScaleCombo->currentText()); QString freq_stop = QStringLiteral("%1%2").arg(1.5*FreqlineEdit->text().toDouble()).arg(FreqScaleCombo->currentText()); s += QStringLiteral("<.SP SP1 1 200 200 0 67 0 0 \"lin\" 1 \"%2\" 1 \"%3\" 1 \"300\" 1 \"no\" 0 \"1\" 0 \"2\" 0>\n").arg((freq_start)).arg((freq_stop)); s += getSPEquationString(50,200); if (microcheck)s += QStringLiteral("\n").arg(Substrate.er).arg(Substrate.height*1e3).arg(Substrate.thickness*1e6).arg(Substrate.tand).arg(Substrate.resistivity).arg(Substrate.roughness); } if (microcheck)//Microstrip implementation { er = Substrate.er; getMicrostrip(Z0, Freq, &Substrate, width, er); s += QStringLiteral("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))); er = Substrate.er; getMicrostrip(Z2, Freq, &Substrate, width, er); s += QStringLiteral("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))); er = Substrate.er; getMicrostrip(Z3, Freq, &Substrate, width, er); s += QStringLiteral("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))); } else { if (LumpedElements)// CLC equivalent { //First capacitor s += QStringLiteral("\n").arg(num2str(CC)); s += QStringLiteral("\n"); //Upper branch s += QStringLiteral("\n").arg(num2str(C2)); s += QStringLiteral("\n"); s += QStringLiteral("\n").arg(num2str(L2)); //Lower branch s += QStringLiteral("\n").arg(num2str(C3)); s += QStringLiteral("\n"); s += QStringLiteral("\n").arg(num2str(L3)); } else { s += QStringLiteral("\n").arg(RoundVariablePrecision(Z0)).arg(ConvertLengthFromM(lambda4)).arg(Alpha);//Z0 line s += QStringLiteral("\n").arg(RoundVariablePrecision(Z2)).arg(ConvertLengthFromM(lambda4)).arg(Alpha);//Output branch 1 s += QStringLiteral("\n").arg(RoundVariablePrecision(Z3)).arg(ConvertLengthFromM(lambda4)).arg(Alpha);//Output branch 2 } } s += QStringLiteral("\n").arg(R);//Isolation resistor if (K!=1) {// An unequal power ratio implies that the load impedance != 50, so it requires matching if (microcheck)//Microstrip { er = Substrate.er; getMicrostrip(sqrt(Z0*R2), Freq, &Substrate, width, er); s += QStringLiteral("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))); er = Substrate.er; getMicrostrip(sqrt(Z0*R3), Freq, &Substrate, width, er); s += QStringLiteral("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))); } else { if (LumpedElements)//CLC equivalent { // Upper branch s += QStringLiteral("\n").arg(num2str(L2_)); s += QStringLiteral("\n").arg(num2str(C2_)); s += QStringLiteral("\n"); // Lower branch s += QStringLiteral("\n").arg(num2str(L3_)); s += QStringLiteral("\n").arg(num2str(C3_)); s += QStringLiteral("\n"); } else { s += QStringLiteral("\n").arg(RoundVariablePrecision(sqrt(Z0*R2))).arg(ConvertLengthFromM(lambda4)).arg(Alpha);//Quarter wave matching output branch 1 s += QStringLiteral("\n").arg(RoundVariablePrecision(sqrt(Z0*R3))).arg(ConvertLengthFromM(lambda4)).arg(Alpha);//Quarter wave matching output branch 2 } } } s += "\n"; //Wiring s += "\n"; s += QStringLiteral("<0 -30 100 -30 \"\" 0 0 0>\n");//Source to Z0 line s += QStringLiteral("<160 -30 190 -30 \"\" 0 0 0 \"\">\n");//Z0 line to branches s += QStringLiteral("<190 -90 190 -30 \"\" 0 0 0 \"\">\n");//Z0 line to upper branch s += QStringLiteral("<190 -90 230 -90 \"\" 0 0 0 \"\">\n");//Z0 line (corner) to upper branch s += QStringLiteral("<290 -90 340 -90 \"\" 0 0 0 \"\">\n");//Upper branch to isolation resistor s += QStringLiteral("<340 -90 340 -50 \"\" 0 0 0 \"\">\n");//Upper branch (corner) to isolation resistor s += QStringLiteral("<190 -30 190 30 \"\" 0 0 0 \"\">\n");//Z0 line to lower branch s += QStringLiteral("<190 30 230 30 \"\" 0 0 0 \"\">\n");//Z0 line (corner) to lower branch s += QStringLiteral("<290 30 340 30 \"\" 0 0 0 \"\">\n");//Lower branch to isolation resistor s += QStringLiteral("<340 10 340 30 \"\" 0 0 0 \"\">\n");//Lower branch (corner) to isolation resistor if (LumpedElements) { s += QStringLiteral("<90 -30 180 -30 \"\" 0 0 0>\n"); s += QStringLiteral("<310 -110 310 -90 \"\" 0 0 0>\n");//Upper branch s += QStringLiteral("<310 30 310 50 \"\" 0 0 0>\n");//Lower branch } if (K!=1)//Unequal power split ratio => need additional matching { if (LumpedElements) { s += QStringLiteral("<340 -90 380 -90 \"\" 0 0 0 \"\">\n");//Upper branch, R to L s += QStringLiteral("<440 -90 500 -90 \"\" 0 0 0 \"\">\n");//Upper branch, L to port s += QStringLiteral("<450 -110 450 -90 \"\" 0 0 0 \"\">\n");//Upper branch, L to C s += QStringLiteral("<340 30 380 30 \"\" 0 0 0 \"\">\n");//Lower branch, R to L s += QStringLiteral("<440 30 500 30 \"\" 0 0 0 \"\">\n");//Lower branch, L to port s += QStringLiteral("<450 30 450 60 \"\" 0 0 0 \"\">\n");//Lower branch, L to C } else//Transmission lines { s += QStringLiteral("<340 -90 380 -90 \"\" 0 0 0 \"\">\n");//Isolation resistor to matching line. Upper branch s += QStringLiteral("<440 -90 500 -90 \"\" 0 0 0 \"\">\n");//Matching line to port 2. Upper branch s += QStringLiteral("<340 30 380 30 \"\" 0 0 0 \"\">\n");//Isolation resistor to matching line. Lowe branch s += QStringLiteral("<440 30 500 30 \"\" 0 0 0 \"\">\n");//Matching line to port 2. Lower branch } } else//Equal power split ratio { s += QStringLiteral("<340 -90 500 -90 \"\" 0 0 0 \"\">\n");//Branch 2 to Port 2 s += QStringLiteral("<340 30 500 30 \"\" 0 0 0 \"\">\n");//Branch 2 to Port 3 } s += "\n"; QApplication::clipboard()->setText(s, QClipboard::Clipboard);//Copy into clipboard return 0; } //----------------------------------------------------------------------------------- // This function calculates a multistage lambda/4 matching using the Chebyshev weigthing. // See Microwave Engineering. David Pozar. John Wiley and Sons. 4th Edition. Pg 256-261 QString QucsPowerCombiningTool::calcChebyLines(double RL, double Z0, double gamma, int N) { if (N > 7)// So far, it is only available Chebyshev weighting up to 7 sections. // Probably, it makes no sense to use a higher number of sections because of the losses { QMessageBox::warning(0, QObject::tr("Error"), QObject::tr("Chebyshev weighting for N>7 is not available")); return QString(); } QString s; double sec_theta_m;// = cosh((1/(1.*N))*acosh((1/gamma)*fabs((RL-Z0)/(Z0+RL))) ); //double sec_theta_m = cosh((1/(1.*N))*acosh(fabs(log(RL/Z0)/(2*gamma))) ); (fabs(log(RL/Z0)/(2*gamma)) < 1) ? sec_theta_m = 0 : sec_theta_m = cosh((1/(1.*N))*acosh(fabs(log(RL/Z0)/(2*gamma))) ); std::vector w(N,0.0); switch(N)//The weights are calculated by equating the reflection coeffient formula to the N-th Chebyshev polinomial { case 1: w[0] = sec_theta_m; break; case 2: w[0] = sec_theta_m*sec_theta_m; w[1] = 2*(sec_theta_m*sec_theta_m-1); break; case 3: w[0] = pow(sec_theta_m,3); w[1] = 3*(pow(sec_theta_m, 3) - sec_theta_m); w[2] = w[1]; break; case 4: w[0] = pow(sec_theta_m, 4); w[1] = 4*sec_theta_m*sec_theta_m*(sec_theta_m*sec_theta_m-1); w[2] =2*( 1-4*sec_theta_m*sec_theta_m+3*pow(sec_theta_m, 4)); w[3]= w[1]; break; case 5: w[0] = pow(sec_theta_m, 5); w[1] = 5*(pow(sec_theta_m, 5) - pow(sec_theta_m, 3)); w[2] = 10*pow(sec_theta_m, 5) - 15*pow(sec_theta_m, 3) + 5*sec_theta_m; w[3] = w[2]; w[4] = w[1]; break; case 6: w[0] = pow(sec_theta_m, 6); w[1] = 6*pow(sec_theta_m,4)*(sec_theta_m*sec_theta_m - 1); w[2] = 15*pow(sec_theta_m, 6) - 24*pow(sec_theta_m, 4) + 9*sec_theta_m*sec_theta_m; w[3] = 2*(10*pow(sec_theta_m, 6) - 18*pow(sec_theta_m, 4) + 9*sec_theta_m*sec_theta_m - 1); w[4] = w[2]; w[5] = w[1]; break; case 7: w[0] = pow(sec_theta_m, 7); w[1] = 7*pow(sec_theta_m, 5)*(sec_theta_m*sec_theta_m -1); w[2] = 21*pow(sec_theta_m, 7) - 35*pow(sec_theta_m, 5) + 14*pow(sec_theta_m, 3); w[3] = 35*pow(sec_theta_m, 7) - 70*pow(sec_theta_m, 5) + 42*pow(sec_theta_m, 3) -7*sec_theta_m ; w[4] = w[3]; w[5] = w[2]; w[6] = w[1]; break; } double Zaux=Z0, Zi; for (int i = 0; i < N; i++) { (RL gamma, int NStages, double Z0) { double Z_, R, Zaux = Zlines.section(';', NStages-1, NStages-1).toDouble(); QString s; std::complex Zi = 0; for (int i=NStages-1; i>=0;i--) { Z_ = abs(Zaux*(Zi + Zaux*tanh(gamma*L))/(Zaux+Zi*tanh(gamma*L))); Zaux = Zlines.section(';', i-1, i-1).toDouble(); R = Z0*Z_/(Z_ - Z0); Zi = Z0; s +=QStringLiteral("%1;").arg(2*R); } return s; } //------------------------------------------------------------------------------ // This function synthesizes a multistage Wilkinson power divider. // References: // [1] A class of broadband three-port TEM-mode hybrids. Seymour B. Cohn. IEEE // transactions on microwave theory and techniques. vol MTT-16, No 2, February 1968 // The approximation of the isolation resistances was taken from // http://www.mathworks.com/matlabcentral/fileexchange/22996-rf-utilities-v1-2/content/RFutils_M/bwilk.m int QucsPowerCombiningTool::MultistageWilkinson(double Z0, double Freq, int NStages, bool SP_block, bool microcheck, tSubstrate Substrate, double Alpha, bool LumpedElements) { QString Zlines = calcChebyLines(2*Z0, Z0, 0.05, NStages); double er, width; microcheck ? er=Substrate.er : 1; double lambda = SPEED_OF_LIGHT/(Freq); double lambda4=lambda/4, W; double alpha; auto C = new double[NStages]; auto L = new double[NStages]; if (LumpedElements)//CLC pi equivalent calculation { double w = 2*M_PI*Freq; int j = 0; for (int i = NStages-1; i>= 0;i--, j++) { double Zi = Zlines.section(';', i, i).toDouble(); L[j] = Zi/w; C[j] = 1./(L[j]*w*w); } } if (microcheck)//Microstrip implementation { double Rs = sqrt((2*M_PI*Freq*4*M_PI*1e-7)/Substrate.resistivity); getMicrostrip(Z0, Freq, &Substrate, W, er); alpha = Rs/(Z0*W);//Conductor attenuation coefficient in (Np/m) } else { alpha = log(pow(0.1*Alpha, 10));//Alpha is given in dB/m, then it is necessary to convert it into Np/m units } std::complex gamma(alpha, 2*M_PI/lambda);//It is only considered the attenation of the metal conductor since it tends to be much higher than the dielectric QString Risol = calcMultistageWilkinsonIsolators(Zlines, lambda4, gamma, NStages, Z0); QString wirestr = "\n", str; QString s = "\n"; s += "\n"; if (SP_block)//Add S-param simulation { //Source s += QStringLiteral("\n").arg(Z0); s += QStringLiteral("\n"); wirestr +=QStringLiteral("<0 0 0 -30 \"\" 0 0 0>\n");//Vertical wire wirestr +=QStringLiteral("<0 -30 70 -30 \"\" 0 0 0>\n");//Horizontal wire //S-parameter analysis component QString freq_start = QStringLiteral("%1%2").arg((1/NStages)*FreqlineEdit->text().toDouble()).arg(FreqScaleCombo->currentText()); QString freq_stop = QStringLiteral("%1%2").arg((2+1/NStages)*FreqlineEdit->text().toDouble()).arg(FreqScaleCombo->currentText()); s += QStringLiteral("<.SP SP1 1 200 200 0 67 0 0 \"lin\" 1 \"%2\" 1 \"%3\" 1 \"300\" 1 \"no\" 0 \"1\" 0 \"2\" 0>\n").arg(freq_start).arg(freq_stop); str += QStringLiteral("\"S11_dB=dB(S[1,1])\" 1 "); str += QStringLiteral("\"S22_dB=dB(S[2,2])\" 1 "); str += QStringLiteral("\"S33_dB=dB(S[3,3])\" 1 "); str += QStringLiteral("\"S21_dB=dB(S[2,1])\" 1 "); str +=QStringLiteral("\"S31_dB=dB(S[3,1])\" 1 "); s += getSPEquationString(50,200); if (microcheck)s += QStringLiteral("\n").arg(Substrate.er).arg(Substrate.height*1e3).arg(Substrate.thickness*1e6).arg(Substrate.tand).arg(Substrate.resistivity).arg(Substrate.roughness); } int x=100; int spacing = 150;//Spacing between sections double Zi, Ri; if (microcheck)//Microstrip { er = Substrate.er; getMicrostrip(Z0, Freq, &Substrate, width, er); s += QStringLiteral("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))).arg(x); } else { if (LumpedElements)//LC elements. Pi CLC equivalent of a lambda/4 line { //First capacitor s += QStringLiteral("\n").arg(num2str(2*C[0])).arg(x); s += QStringLiteral("\n").arg(x); wirestr +=QStringLiteral("<%1 -30 %2 -30 \"\" 0 0 0>\n").arg(x-30).arg(x+30); } else//Ideal transmission lines { s += QStringLiteral("\n").arg(x).arg(RoundVariablePrecision(Z0)).arg(ConvertLengthFromM(lambda4)).arg(Alpha);//Z0 line } } x+=50;//Separation between the source transmission line and the beginning of the output branches wirestr +=QStringLiteral("<%1 -30 %2 -30 \"\" 0 0 0>\n").arg(x-20).arg(x+30);//Vertical line joining the output branches wirestr +=QStringLiteral("<%1 -90 %1 30 \"\" 0 0 0>\n").arg(x+30);//Vertical line joining the output branches int aux=1; for (int i = NStages-1; i>= 0;i--, aux++) { Zi = Zlines.section(';', i, i).toDouble(); Ri = Risol.section(';', (NStages-1)-i, (NStages-1)-i).toDouble(); wirestr +=QStringLiteral("<%1 30 %2 30 \"\" 0 0 0>\n").arg(x+30).arg(x+70); wirestr +=QStringLiteral("<%1 -90 %2 -90 \"\" 0 0 0>\n").arg(x+30).arg(x+70); wirestr +=QStringLiteral("<%1 30 %2 30 \"\" 0 0 0>\n").arg(x+130).arg(x+180); wirestr +=QStringLiteral("<%1 -90 %2 -90 \"\" 0 0 0>\n").arg(x+130).arg(x+180); //Wiring the isolation resistor wirestr +=QStringLiteral("<%1 30 %1 10 \"\" 0 0 0>\n").arg(x+160); wirestr +=QStringLiteral("<%1 -90 %1 -50 \"\" 0 0 0>\n").arg(x+160); if (microcheck) { er = Substrate.er; getMicrostrip(Zi, Freq, &Substrate, width, er); s += QStringLiteral("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))).arg(x+100); er = Substrate.er; getMicrostrip(Zi, Freq, &Substrate, width, er); s += QStringLiteral("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))).arg(x+100); } else { if (LumpedElements)//Lumped equivalent { if (i == 0)//Last element { // Upper branch s += QStringLiteral("\n").arg(num2str(L[aux-1])).arg(x+100); s += QStringLiteral("\n").arg(num2str(C[aux-1])).arg(x+160); wirestr +=QStringLiteral("<%1 -90 %1 -120 \"\" 0 0 0>\n").arg(x+160); s += QStringLiteral("\n").arg(x+160); // Lower branch s += QStringLiteral("\n").arg(num2str(L[aux-1])).arg(x+100); s += QStringLiteral("\n").arg(num2str(C[aux-1])).arg(x+160); wirestr +=QStringLiteral("<%1 30 %1 60 \"\" 0 0 0>\n").arg(x+160); s += QStringLiteral("\n").arg(x+160); } else { // Upper branch s += QStringLiteral("\n").arg(num2str(L[aux-1])).arg(x+100); s += QStringLiteral("\n").arg(num2str(C[aux]+C[aux-1])).arg(x+160); wirestr +=QStringLiteral("<%1 -90 %1 -120 \"\" 0 0 0>\n").arg(x+160); s += QStringLiteral("\n").arg(x+160); // Lower branch s += QStringLiteral("\n").arg(num2str(L[aux-1])).arg(x+100); s += QStringLiteral("\n").arg(num2str(C[aux]+C[aux-1])).arg(x+160); wirestr +=QStringLiteral("<%1 30 %1 60 \"\" 0 0 0>\n").arg(x+160); s += QStringLiteral("\n").arg(x+160); } } else//Ideal transmission lines { s += QStringLiteral("\n").arg(x+100).arg(RoundVariablePrecision(Zi)).arg(ConvertLengthFromM(lambda4)).arg(Alpha);//Upper branch s += QStringLiteral("\n").arg(x+100).arg(RoundVariablePrecision(Zi)).arg(ConvertLengthFromM(lambda4)).arg(Alpha);//Lower branch } } s += QStringLiteral("\n").arg(x+160).arg(RoundVariablePrecision(Ri));//Isolation resistor x+=spacing; if(SP_block && (i==1))//Add output ports at the last stage { s += QStringLiteral("\n").arg(x+300).arg(Z0); s += QStringLiteral("\n").arg(x+300); wirestr +=QStringLiteral("<%1 30 %2 30 \"\" 0 0 0>\n").arg(x+150).arg(x+300); s += QStringLiteral("\n").arg(x+300).arg(Z0); s += QStringLiteral("\n").arg(x+300); wirestr +=QStringLiteral("<%1 -90 %2 -90 \"\" 0 0 0>\n").arg(x+150).arg(x+300); } } s += "\n"; wirestr+="\n";; s += wirestr; QApplication::clipboard()->setText(s, QClipboard::Clipboard); return 0; } //------------------------------------------------------------------------------ // This function generates the schematic of a tee power divider // Reference: "High Efficiency RF and Microwave Solid State Power Amplifiers". Paolo Colantonio, Franco Giannini and Ernesto Limiti, 2009, Wiley int QucsPowerCombiningTool::Tee(double Z0, double Freq, double K, bool SP_block, bool microcheck, tSubstrate Substrate, double Alpha) { double er, width; double lambda4=SPEED_OF_LIGHT/(4*Freq); QString s = "\n"; s += "\n"; if (SP_block) { //Source s += QStringLiteral("\n").arg(Z0); s += QStringLiteral("\n"); //Output port 1 s += QStringLiteral("\n").arg(2*Z0); s += QStringLiteral("\n"); //Output port 2 s += QStringLiteral("\n").arg(2*Z0); s += QStringLiteral("\n"); //S-parameter analysis component QString freq_start = QStringLiteral("%1%2").arg(0.5*FreqlineEdit->text().toDouble()).arg(FreqScaleCombo->currentText()); QString freq_stop = QStringLiteral("%1%2").arg(1.5*FreqlineEdit->text().toDouble()).arg(FreqScaleCombo->currentText()); s += QStringLiteral("<.SP SP1 1 200 200 0 67 0 0 \"lin\" 1 \"%2\" 1 \"%3\" 1 \"300\" 1 \"no\" 0 \"1\" 0 \"2\" 0>\n").arg(freq_start).arg(freq_stop); s += QStringLiteral("\n"); if (microcheck)s += QStringLiteral("\n").arg(Substrate.er).arg(Substrate.height*1e3).arg(Substrate.thickness*1e6).arg(Substrate.tand).arg(Substrate.resistivity).arg(Substrate.roughness); } if (microcheck) { er = Substrate.er; getMicrostrip(Z0, Freq, &Substrate, width, er); s += QStringLiteral("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))); er = Substrate.er; getMicrostrip(Z0*(K+1), Freq, &Substrate, width, er); s += QStringLiteral("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))); er = Substrate.er; getMicrostrip(Z0*(K+1)/K, Freq, &Substrate, width, er); s += QStringLiteral("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))); } else { s += QStringLiteral("\n").arg(RoundVariablePrecision(Z0)).arg(ConvertLengthFromM(lambda4)).arg(Alpha);//Z0 line s += QStringLiteral("\n").arg(RoundVariablePrecision(Z0*(K+1))).arg(ConvertLengthFromM(lambda4)).arg(Alpha);//Output branch 1 s += QStringLiteral("\n").arg(RoundVariablePrecision(Z0*(K+1)/K)).arg(ConvertLengthFromM(lambda4)).arg(Alpha);//Output branch 2 } if (K!=1) {// An unequal power ratio implies that the load impedance != 50, so it requires matching if (microcheck) { er = Substrate.er; getMicrostrip(sqrt(2*Z0*Z0*(K+1)), Freq, &Substrate, width, er); s += QStringLiteral("\n").arg(width).arg(lambda4/sqrt(er)); er = Substrate.er; getMicrostrip(sqrt(Z0*Z0*(K+1)/K), Freq, &Substrate, width, er); s += QStringLiteral("\n").arg(width).arg(lambda4/sqrt(er)); } else { s += QStringLiteral("\n").arg(RoundVariablePrecision(sqrt(2*Z0*Z0*(K+1)))).arg(ConvertLengthFromM(lambda4)).arg(Alpha);//Quarter wave matching output branch 1 s += QStringLiteral("\n").arg(RoundVariablePrecision(sqrt(2*Z0*Z0*(K+1)/K))).arg(ConvertLengthFromM(lambda4)).arg(Alpha);//Quarter wave matching output branch 2 } } s += "\n"; //Wiring s += "\n"; s += QStringLiteral("<0 -30 90 -30 \"\" 0 0 0>\n");//Source to Z0 line s += QStringLiteral("<150 -30 200 -30 \"\" 0 0 0>\n");//Source to Z0 line s += QStringLiteral("<200 30 200 -90 \"\" 0 0 0>\n");//Z0 line to branches s += QStringLiteral("<200 30 240 30 \"\" 0 0 0>\n");//Z0 line to branch 1 s += QStringLiteral("<200 -90 240 -90 \"\" 0 0 0>\n");//Z0 line to branch 2 s += QStringLiteral("<300 30 340 30 \"\" 0 0 0>\n");//Branch 2to R s += QStringLiteral("<300 -90 340 -90 \"\" 0 0 0>\n");//Branch 1 to R if (K!=1) { s += QStringLiteral("<400 30 450 30 \"\" 0 0 0>\n");//Branch 2 to Port 2 s += QStringLiteral("<400 -90 450 -90 \"\" 0 0 0>\n");//Branch 2 to Port 3 } else { s += QStringLiteral("<340 30 450 30 \"\" 0 0 0>\n");//Branch 2 to Port 2 s += QStringLiteral("<340 -90 450 -90 \"\" 0 0 0>\n");//Branch 2 to Port 3 } s += "\n"; QApplication::clipboard()->setText(s, QClipboard::Clipboard); return 0; } //-------------------------------------------------------------------------------- // This function generates the schematic of a Branch-line coupler int QucsPowerCombiningTool::Branchline(double Z0, double Freq, double K, bool SP_block, bool microcheck, tSubstrate Substrate, double Alpha) { double er, width; double lambda4=SPEED_OF_LIGHT/(4*Freq); double ZA = Z0*sqrt(K/(K+1)); double ZB = Z0*sqrt(K); QString s = "\n"; s += "\n"; if (SP_block) { //Source s += QStringLiteral("\n").arg(Z0); s += QStringLiteral("\n"); //Output port 1 s += QStringLiteral("\n").arg(Z0); s += QStringLiteral("\n"); //Output port 2 s += QStringLiteral("\n").arg(Z0); s += QStringLiteral("\n"); //S-parameter analysis component QString freq_start = QStringLiteral("%1%2").arg(0.5*FreqlineEdit->text().toDouble()).arg(FreqScaleCombo->currentText()); QString freq_stop = QStringLiteral("%1%2").arg(1.5*FreqlineEdit->text().toDouble()).arg(FreqScaleCombo->currentText()); s += QStringLiteral("<.SP SP1 1 200 200 0 67 0 0 \"lin\" 1 \"%2\" 1 \"%3\" 1 \"300\" 1 \"no\" 0 \"1\" 0 \"2\" 0>\n").arg(freq_start).arg(freq_stop); s += QStringLiteral("\n"); if (microcheck)s += QStringLiteral("\n").arg(Substrate.er).arg(Substrate.height*1e3).arg(Substrate.thickness*1e6).arg(Substrate.tand).arg(Substrate.resistivity).arg(Substrate.roughness); } //Branch line coupler if (microcheck) { er = Substrate.er; getMicrostrip(ZA, Freq, &Substrate, width, er); s += QStringLiteral("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))); er = Substrate.er; getMicrostrip(ZA, Freq, &Substrate, width, er); s += QStringLiteral("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))); er = Substrate.er; getMicrostrip(ZB, Freq, &Substrate, width, er); s += QStringLiteral("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))); er = Substrate.er; getMicrostrip(ZB, Freq, &Substrate, width, er); s += QStringLiteral("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))); } else { s += QStringLiteral("\n").arg(RoundVariablePrecision(ZA)).arg(ConvertLengthFromM(lambda4)).arg(Alpha);//Z0 line s += QStringLiteral("\n").arg(RoundVariablePrecision(ZA)).arg(ConvertLengthFromM(lambda4)).arg(Alpha);//Output branch 1 s += QStringLiteral("\n").arg(RoundVariablePrecision(ZB)).arg(ConvertLengthFromM(lambda4)).arg(Alpha);//Output branch 2 s += QStringLiteral("\n").arg(RoundVariablePrecision(ZB)).arg(ConvertLengthFromM(lambda4)).arg(Alpha);//Output branch 2 } s += QStringLiteral("\n").arg(Z0);//Isolated port s += QStringLiteral("\n"); s += "\n"; //Wiring s += "\n"; s += QStringLiteral("<150 -90 190 -90 \"\" 0 0 0 \"\">\n"); s += QStringLiteral("<150 -90 150 -50 \"\" 0 0 0 \"\">\n"); s += QStringLiteral("<290 -90 290 -50 \"\" 0 0 0 \"\">\n"); s += QStringLiteral("<250 -90 290 -90 \"\" 0 0 0 \"\">\n"); s += QStringLiteral("<250 30 290 30 \"\" 0 0 0 \"\">\n"); s += QStringLiteral("<290 10 290 30 \"\" 0 0 0 \"\">\n"); s += QStringLiteral("<150 30 190 30 \"\" 0 0 0 \"\">\n"); s += QStringLiteral("<150 10 150 30 \"\" 0 0 0 \"\">\n"); s += QStringLiteral("<290 -150 290 -90 \"\" 0 0 0 \"\">\n"); s += QStringLiteral("<290 -150 400 -150 \"\" 0 0 0 \"\">\n"); s += QStringLiteral("<150 -150 150 -90 \"\" 0 0 0 \"\">\n"); s += QStringLiteral("<50 -150 150 -150 \"\" 0 0 0 \"\">\n"); s += QStringLiteral("<150 30 150 60 \"\" 0 0 0 \"\">\n"); s += QStringLiteral("<150 30 150 60 \"\" 0 0 0 \"\">\n"); s += QStringLiteral("<50 60 150 60 \"\" 0 0 0 \"\">\n"); s += QStringLiteral("<290 30 290 60 \"\" 0 0 0 \"\">\n"); s += QStringLiteral("<290 60 400 60 \"\" 0 0 0 \"\">\n"); s += "\n"; QApplication::clipboard()->setText(s, QClipboard::Clipboard); return 0; } //-------------------------------------------------------------------------------- // This function generates the schematic of a double box branchline coupler. // Reference: Kumar, S.; Danshin, Tom, "A Multisection Broadband Impedance Transforming Branchline Quad Hybrid Suitable for MMIC Realization," // in Microwave Conference, 1992. 22nd European , vol.2, no., pp.1301-1306, 5-9 Sept. 1992 int QucsPowerCombiningTool::DoubleBoxBranchline(double Z0, double Freq, double K, bool SP_block, bool microcheck, tSubstrate Substrate, double Alpha) { double er, width; double lambda4=SPEED_OF_LIGHT/(4*Freq); double r=1; double t = sqrt((1+K)*r); double ZA = Z0*sqrt(r*(t*t -r))/(t-r); double ZD = Z0*sqrt(r*(t*t -r))/(t-1); double ZB = Z0*sqrt(r-(r*r)/(t*t)); QString s = "\n"; s += "\n"; if (SP_block) { //Source s += QStringLiteral("\n").arg(Z0); s += QStringLiteral("\n"); //Output port 1 s += QStringLiteral("\n").arg(Z0); s += QStringLiteral("\n"); //Output port 2 s += QStringLiteral("\n").arg(Z0); s += QStringLiteral("\n"); //S-parameter analysis component QString freq_start = QStringLiteral("%1%2").arg(0.5*FreqlineEdit->text().toDouble()).arg(FreqScaleCombo->currentText()); QString freq_stop = QStringLiteral("%1%2").arg(1.5*FreqlineEdit->text().toDouble()).arg(FreqScaleCombo->currentText()); s += QStringLiteral("<.SP SP1 1 200 200 0 67 0 0 \"lin\" 1 \"%2\" 1 \"%3\" 1 \"300\" 1 \"no\" 0 \"1\" 0 \"2\" 0>\n").arg(freq_start).arg(freq_stop); s += QStringLiteral("\n"); if (microcheck)s += QStringLiteral("\n").arg(Substrate.er).arg(Substrate.height*1e3).arg(Substrate.thickness*1e6).arg(Substrate.tand).arg(Substrate.resistivity).arg(Substrate.roughness); } //Branch line coupler if (microcheck) { er = Substrate.er; getMicrostrip(ZB, Freq, &Substrate, width, er); s += QStringLiteral("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))); er = Substrate.er; getMicrostrip(ZB, Freq, &Substrate, width, er); s += QStringLiteral("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))); er = Substrate.er; getMicrostrip(ZB, Freq, &Substrate, width, er); s += QStringLiteral("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))); er = Substrate.er; getMicrostrip(ZB, Freq, &Substrate, width, er); s += QStringLiteral("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))); er = Substrate.er; getMicrostrip(ZB, Freq, &Substrate, width, er); s += QStringLiteral("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))); er = Substrate.er; getMicrostrip(ZA, Freq, &Substrate, width, er); s += QStringLiteral("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))); er = Substrate.er; getMicrostrip(ZD, Freq, &Substrate, width, er); s += QStringLiteral("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))); } else { s += QStringLiteral("\n").arg(RoundVariablePrecision(ZB)).arg(ConvertLengthFromM(lambda4)).arg(Alpha); s += QStringLiteral("\n").arg(RoundVariablePrecision(ZB)).arg(ConvertLengthFromM(lambda4)).arg(Alpha); s += QStringLiteral("\n").arg(RoundVariablePrecision(ZB)).arg(ConvertLengthFromM(lambda4)).arg(Alpha); s += QStringLiteral("\n").arg(RoundVariablePrecision(ZB)).arg(ConvertLengthFromM(lambda4)).arg(Alpha); s += QStringLiteral("\n").arg(RoundVariablePrecision(ZB)).arg(ConvertLengthFromM(lambda4)).arg(Alpha); s += QStringLiteral("\n").arg(RoundVariablePrecision(ZA)).arg(ConvertLengthFromM(lambda4)).arg(Alpha); s += QStringLiteral("\n").arg(RoundVariablePrecision(ZD)).arg(ConvertLengthFromM(lambda4)).arg(Alpha); } s += QStringLiteral("\n").arg(Z0);//Isolated port s += QStringLiteral("\n"); s += "\n"; //Wiring s += "\n"; s += QStringLiteral("<290 -90 290 -50 \"\" 0 0 0 \"\">\n"); s += QStringLiteral("<290 10 290 30 \"\" 0 0 0 \"\">\n"); s += QStringLiteral("<290 -90 320 -90 \"\" 0 0 0 \"\">\n"); s += QStringLiteral("<250 -90 290 -90 \"\" 0 0 0 \"\">\n"); s += QStringLiteral("<290 30 320 30 \"\" 0 0 0 \"\">\n"); s += QStringLiteral("<250 30 290 30 \"\" 0 0 0 \"\">\n"); s += QStringLiteral("<380 -90 420 -90 \"\" 0 0 0 \"\">\n"); s += QStringLiteral("<420 -90 420 -50 \"\" 0 0 0 \"\">\n"); s += QStringLiteral("<380 30 420 30 \"\" 0 0 0 \"\">\n"); s += QStringLiteral("<420 10 420 30 \"\" 0 0 0 \"\">\n"); s += QStringLiteral("<40 30 40 60 \"\" 0 0 0 \"\">\n"); s += QStringLiteral("<40 30 150 30 \"\" 0 0 0 \"\">\n"); s += QStringLiteral("<150 30 190 30 \"\" 0 0 0 \"\">\n"); s += QStringLiteral("<150 10 150 30 \"\" 0 0 0 \"\">\n"); s += QStringLiteral("<150 -90 150 -50 \"\" 0 0 0 \"\">\n"); s += QStringLiteral("<150 -90 190 -90 \"\" 0 0 0 \"\">\n"); s += QStringLiteral("<420 -160 420 -90 \"\" 0 0 0 \"\">\n"); s += QStringLiteral("<420 -160 500 -160 \"\" 0 0 0 \"\">\n"); s += QStringLiteral("<500 -160 500 -130 \"\" 0 0 0 \"\">\n"); s += QStringLiteral("<500 30 500 60 \"\" 0 0 0 \"\">\n"); s += QStringLiteral("<420 30 500 30 \"\" 0 0 0 \"\">\n"); s += QStringLiteral("<130 -160 150 -90 \"\" 0 0 0 \"\">\n"); s += QStringLiteral("<40 -160 150 -160 \"\" 0 0 0 \"\">\n"); s += QStringLiteral("<40 -160 40 -130 \"\" 0 0 0 \"\">\n"); s += "\n"; QApplication::clipboard()->setText(s, QClipboard::Clipboard); return 0; } //----------------------------------------------------------------------------- // This function generates the schematic of a N-way Bagley combiner (N odd) // Reference: "High Efficiency RF and Microwave Solid State Power Amplifiers". Paolo Colantonio, Franco Giannini and Ernesto Limiti, 2009, Wiley. Pg. 411 int QucsPowerCombiningTool::Bagley(double Z0, double Freq, int N, bool SP_block, bool microcheck, tSubstrate Substrate, double Alpha) { if (N % 2 == 0) { N++; QString str = QStringLiteral("The number of outputs must be an odd number. N=%1 will be used instead").arg(N); QMessageBox::warning(this, tr("Bagley"), str, QMessageBox::Close); } double er, width; double lambda4=SPEED_OF_LIGHT/(4*Freq); double lambda2=lambda4*2; double Zbranch = 2*Z0/sqrt(N); QString s = "\n"; s += "\n"; if (SP_block) { //Source s += QStringLiteral("\n").arg(Z0); s += QStringLiteral("\n"); QString freq_start = QStringLiteral("%1%2").arg(0.5*FreqlineEdit->text().toDouble()).arg(FreqScaleCombo->currentText()); QString freq_stop = QStringLiteral("%1%2").arg(1.5*FreqlineEdit->text().toDouble()).arg(FreqScaleCombo->currentText()); s += QStringLiteral("<.SP SP1 1 200 200 0 67 0 0 \"lin\" 1 \"%2\" 1 \"%3\" 1 \"300\" 1 \"no\" 0 \"1\" 0 \"2\" 0>\n").arg(freq_start).arg(freq_stop); //Equations QString str = QStringLiteral(" \"S11_dB=dB(S[1,1])\" 1 "); for (int i=2;i<=N+1; i++) { str += QStringLiteral("\"S%1%2_dB=dB(S[%1,1])\" 1 ").arg(i).arg(1); str += QStringLiteral("\"S%1%1_dB=dB(S[%1,%1])\" 1 ").arg(i); } s += QStringLiteral("\n"); if (microcheck)s += QStringLiteral("\n").arg(Substrate.er).arg(Substrate.height*1e3).arg(Substrate.thickness*1e6).arg(Substrate.tand).arg(Substrate.resistivity).arg(Substrate.roughness); } //Input section if (microcheck) { er = Substrate.er; getMicrostrip(Zbranch, Freq, &Substrate, width, er); s += QStringLiteral("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))); er = Substrate.er; getMicrostrip(Zbranch, Freq, &Substrate, width, er); s += QStringLiteral("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))); } else { s += QStringLiteral("\n").arg(RoundVariablePrecision(Zbranch)).arg(ConvertLengthFromM(lambda4)).arg(Alpha); s += QStringLiteral("\n").arg(RoundVariablePrecision(Zbranch)).arg(ConvertLengthFromM(lambda4)).arg(Alpha); } //Output branches int x = 240; s += QStringLiteral("\n").arg(x-100).arg(Z0); s += QStringLiteral("\n").arg(x-100); for (int i=1;i\n").arg(x).arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda2/sqrt(er))); } else { s += QStringLiteral("\n").arg(x).arg(RoundVariablePrecision(Zbranch)).arg(ConvertLengthFromM(lambda2)).arg(Alpha); } x+=100; if (SP_block) { //i-th output port s += QStringLiteral("\n").arg(x).arg(Z0); s += QStringLiteral("\n").arg(x); } x+=100; } s += "\n"; //Wiring s += "\n"; s += QStringLiteral("<0 -90 100 -90 \"\" 0 0 0>\n");//Source to lambda/4 lines s += QStringLiteral("<100 -110 100 -60 \"\" 0 0 0>\n");//lambda/4 lines s += QStringLiteral("<100 30 140 30 \"\" 0 0 0>\n");//Lower lambda/4 line to the first lambda/2 section s += QStringLiteral("<100 30 100 0 \"\" 0 0 0>\n");//Lower lambda/4 line to the first lambda/2 section //Wiring the rest of the lambda/2 sections x=140; for (int i=1;i\n").arg(x).arg(x+70); s += QStringLiteral("<%1 30 %2 30 \"\" 0 0 0>\n").arg(x+130).arg(x+200); x+=200; } s += QStringLiteral("<%1 30 %2 30 \"\" 0 0 0>\n").arg(x).arg(x+30); s += QStringLiteral("<%1 30 %1 -200 \"\" 0 0 0>\n").arg(x+30);//Final lambda/2 section to the upper lambda/4 section. Vertical line s += QStringLiteral("<100 -200 %1 -200 \"\" 0 0 0>\n").arg(x+30);//Final lambda/2 section to the upper lambda/4 section. Horizontal line s += QStringLiteral("<100 -170 100 -200 \"\" 0 0 0>\n"); s += "\n"; QApplication::clipboard()->setText(s, QClipboard::Clipboard); return 0; } //--------------------------------------------------------------------------- // This function generates a Gysel combiner int QucsPowerCombiningTool::Gysel(double Z0, double Freq, bool SP_block, bool microcheck, tSubstrate Substrate, double Alpha) { double er, width; double lambda4=SPEED_OF_LIGHT/(4*Freq); QString s = "\n"; s += "\n"; if (SP_block) { //Source s += QStringLiteral("\n").arg(Z0); s += QStringLiteral("\n"); //Output port 1 s += QStringLiteral("\n").arg(Z0); s += QStringLiteral("\n"); //Output port 2 s += QStringLiteral("\n").arg(Z0); s += QStringLiteral("\n"); //S-parameter analysis component QString freq_start = QStringLiteral("%1%2").arg(0.5*FreqlineEdit->text().toDouble()).arg(FreqScaleCombo->currentText()); QString freq_stop = QStringLiteral("%1%2").arg(1.5*FreqlineEdit->text().toDouble()).arg(FreqScaleCombo->currentText()); s += QStringLiteral("<.SP SP1 1 200 200 0 67 0 0 \"lin\" 1 \"%2\" 1 \"%3\" 1 \"300\" 1 \"no\" 0 \"1\" 0 \"2\" 0>\n").arg(freq_start).arg(freq_stop); s += QStringLiteral("\n"); if (microcheck)s += QStringLiteral("\n").arg(Substrate.er).arg(Substrate.height*1e3).arg(Substrate.thickness*1e6).arg(Substrate.tand).arg(Substrate.resistivity).arg(Substrate.roughness); } if (microcheck) { er = Substrate.er; getMicrostrip(sqrt(2)*Z0, Freq, &Substrate, width, er); s += QStringLiteral("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))); er = Substrate.er; getMicrostrip(sqrt(2)*Z0, Freq, &Substrate, width, er); s += QStringLiteral("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))); er = Substrate.er; getMicrostrip(Z0, Freq, &Substrate, width, er); s += QStringLiteral("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))); er = Substrate.er; getMicrostrip(Z0, Freq, &Substrate, width, er); s += QStringLiteral("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))); er = Substrate.er; getMicrostrip(Z0/sqrt(2), Freq, &Substrate, width, er); s += QStringLiteral("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(2*lambda4/sqrt(er))); } else { s += QStringLiteral("\n").arg(RoundVariablePrecision(sqrt(2)*Z0)).arg(ConvertLengthFromM(lambda4)).arg(Alpha); s += QStringLiteral("\n").arg(RoundVariablePrecision(sqrt(2)*Z0)).arg(ConvertLengthFromM(lambda4)).arg(Alpha); s += QStringLiteral("\n").arg(Z0).arg(ConvertLengthFromM(lambda4)).arg(Alpha); s += QStringLiteral("\n").arg(Z0).arg(ConvertLengthFromM(lambda4)).arg(Alpha); s += QStringLiteral("\n").arg(RoundVariablePrecision(Z0/sqrt(2))).arg(ConvertLengthFromM(2*lambda4)).arg(Alpha); } //Resistors s += QStringLiteral("\n").arg(Z0);//Isolation resistor s += QStringLiteral("\n"); s += QStringLiteral("\n").arg(Z0);//Isolation resistor s += QStringLiteral("\n"); s += "\n"; s += "\n"; s += QStringLiteral("<120 70 120 100 \"\" 0 0 0 \"\">\n");//Source to the lines at the input s += QStringLiteral("<120 -130 120 -100 \"\" 0 0 0 \"\">\n");//Line between the two lines at the input s += QStringLiteral("<120 -130 190 -130 \"\" 0 0 0 \"\">\n"); s += QStringLiteral("<120 -200 120 -130 \"\" 0 0 0 \"\">\n"); s += QStringLiteral("<250 -130 320 -130 \"\" 0 0 0 \"\">\n"); s += QStringLiteral("<320 -130 320 -50 \"\" 0 0 0 \"\">\n"); s += QStringLiteral("<120 100 190 100 \"\" 0 0 0 \"\">\n");//Line between the line on the top to the upper resistor s += QStringLiteral("<250 100 320 100 \"\" 0 0 0 \"\">\n");//Line between the line on the top to the lower resistor s += QStringLiteral("<320 10 320 100 \"\" 0 0 0 \"\">\n"); s += QStringLiteral("<30 -200 120 -200 \"\" 0 0 0 \"\">\n"); s += QStringLiteral("<30 -200 30 -190 \"\" 0 0 0 \"\">\n"); s += QStringLiteral("<400 -200 400 -190 \"\" 0 0 0 \"\">\n"); s += QStringLiteral("<320 -200 320 -130 \"\" 0 0 0 \"\">\n"); s += QStringLiteral("<320 -200 400 -200 \"\" 0 0 0 \"\">\n"); s += QStringLiteral("<320 100 400 100 \"\" 0 0 0 \"\">\n"); s += QStringLiteral("<30 100 120 100 \"\" 0 0 0 \"\">\n"); s += QStringLiteral("<30 100 30 110 \"\" 0 0 0 \"\">\n"); s += QStringLiteral("<120 -40 120 -30 \"\" 0 0 0 \"\">\n"); s += QStringLiteral("<120 -30 120 10 \"\" 0 0 0 \"\">\n"); s += QStringLiteral("<0 -30 120 -30 \"\" 0 0 0 \"\">\n"); s += "\n"; QApplication::clipboard()->setText(s, QClipboard::Clipboard); return 0; } int QucsPowerCombiningTool::TravellingWave(double Z0, double Freq, int N, bool SP_block, bool microcheck, tSubstrate Substrate, double Alpha) { double er, width; double lambda4=SPEED_OF_LIGHT/(4*Freq); QString wirestr = "\n", str; QString s = "\n"; s += "\n"; if (SP_block) { //Source s += QStringLiteral("\n").arg(Z0); s += QStringLiteral("\n"); wirestr +=QStringLiteral("<0 0 0 -30 \"\" 0 0 0>\n");//Vertical wire wirestr +=QStringLiteral("<0 -30 40 -30 \"\" 0 0 0>\n");//Horizontal wire //S-parameter analysis component QString freq_start = QStringLiteral("%1%2").arg(0.5*FreqlineEdit->text().toDouble()).arg(FreqScaleCombo->currentText()); QString freq_stop = QStringLiteral("%1%2").arg(1.5*FreqlineEdit->text().toDouble()).arg(FreqScaleCombo->currentText()); s += QStringLiteral("<.SP SP1 1 200 200 0 67 0 0 \"lin\" 1 \"%2\" 1 \"%3\" 1 \"300\" 1 \"no\" 0 \"1\" 0 \"2\" 0>\n").arg(freq_start).arg(freq_stop); // Equations str = QStringLiteral("\"S11_dB=dB(S[1,1])\" 1 "); for (int i=2;i<=N+1; i++) { str += QStringLiteral("\"S%1%2_dB=dB(S[%1,1])\" 1 ").arg(i).arg(1); str += QStringLiteral("\"S%1%1_dB=dB(S[%1,%1])\" 1 ").arg(i); } s += QStringLiteral("\n"); if (microcheck)s += QStringLiteral("\n").arg(Substrate.er).arg(Substrate.height*1e3).arg(Substrate.thickness*1e6).arg(Substrate.tand).arg(Substrate.resistivity).arg(Substrate.roughness); } QString wilkstr, aux_str, aux_str_2; double Z2, Z3, R, R2, R3; int x=100; int spacing = 350; for (int n = N-1; n>0;n--) { wilkstr = CalculateWilkinson(Z0, sqrt(n)); Z2 = wilkstr.section(';', 0, 0).toDouble(); Z3 = wilkstr.section(';', 1, 1).toDouble(); R = wilkstr.section(';', 2, 2).toDouble(); R2 = wilkstr.section(';', 3, 3).toDouble(); R3 = wilkstr.section(';', 4, 4).toDouble(); wirestr +=QStringLiteral("<%1 -30 %2 -30 \"\" 0 0 0>\n").arg(x-60).arg(x-30); wirestr +=QStringLiteral("<%1 -30 %2 -30 \"\" 0 0 0>\n").arg(x+30).arg(x+60); wirestr +=QStringLiteral("<%1 30 %2 30 \"\" 0 0 0>\n").arg(x+60).arg(x+100); wirestr +=QStringLiteral("<%1 -90 %2 -90 \"\" 0 0 0>\n").arg(x+60).arg(x+100); wirestr +=QStringLiteral("<%1 -90 %1 30 \"\" 0 0 0>\n").arg(x+60); wirestr +=QStringLiteral("<%1 30 %2 30 \"\" 0 0 0>\n").arg(x+160).arg(x+200); wirestr +=QStringLiteral("<%1 -90 %2 -90 \"\" 0 0 0>\n").arg(x+160).arg(x+200); wirestr +=QStringLiteral("<%1 30 %2 30 \"\" 0 0 0>\n").arg(x+260).arg(x+350); wirestr +=QStringLiteral("<%1 -90 %2 -90 \"\" 0 0 0>\n").arg(x+260).arg(x+290); wirestr +=QStringLiteral("<%1 30 %1 60 \"\" 0 0 0>\n").arg(x+350); //Wiring the isolation resistor wirestr +=QStringLiteral("<%1 30 %1 10 \"\" 0 0 0>\n").arg(x+190); wirestr +=QStringLiteral("<%1 -90 %1 -50 \"\" 0 0 0>\n").arg(x+190); if(n>1)wirestr +=QStringLiteral("<%1 -90 %1 -30 \"\" 0 0 0>\n").arg(x+290); if(n==1)wirestr +=QStringLiteral("<%1 -90 %2 -90 \"\" 0 0 0>\n").arg(x+290).arg(x+350);//Last power combiner if (microcheck) { er = Substrate.er; getMicrostrip(Z0, Freq, &Substrate, width, er); s += QStringLiteral("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))).arg(x); er = Substrate.er; getMicrostrip(Z3, Freq, &Substrate, width, er); s += QStringLiteral("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))).arg(x+130); er = Substrate.er; getMicrostrip(Z2, Freq, &Substrate, width, er); s += QStringLiteral("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))).arg(x+130); } else { aux_str = ConvertLengthFromM(lambda4); s += QStringLiteral("\n").arg(x).arg(Z0).arg(aux_str).arg(Alpha);//Z0 line s += QStringLiteral("\n").arg(x+130).arg(RoundVariablePrecision(Z3)).arg(aux_str).arg(Alpha);//Output branch 1 s += QStringLiteral("\n").arg(x+130).arg(RoundVariablePrecision(Z2)).arg(aux_str).arg(Alpha);//Output branch 2 } s += QStringLiteral("\n").arg(x+190).arg(RoundVariablePrecision(R));//Isolation resistor if ((R2!=50)||(R3 != 50)) {// An unequal power ratio implies that the load impedance != 50, so it requires matching. if (microcheck) { er = Substrate.er; getMicrostrip(sqrt(Z0*R3), Freq, &Substrate, width, er); aux_str = ConvertLengthFromM(lambda4/sqrt(er)); aux_str_2 = ConvertLengthFromM(width); s += QStringLiteral("\n").arg(aux_str_2).arg(aux_str).arg(x+230); er = Substrate.er; getMicrostrip(sqrt(Z0*R2), Freq, &Substrate, width, er); aux_str = ConvertLengthFromM(lambda4/sqrt(er)); aux_str_2 = ConvertLengthFromM(width); s += QStringLiteral("\n").arg(aux_str_2).arg(aux_str).arg(x+230); } else { s += QStringLiteral("\n").arg(x+230).arg(RoundVariablePrecision(sqrt(Z0*R3))).arg(ConvertLengthFromM(lambda4)).arg(Alpha);//Quarter wave matching output branch 1 s += QStringLiteral("\n").arg(x+230).arg(RoundVariablePrecision(sqrt(Z0*R2))).arg(ConvertLengthFromM(lambda4)).arg(Alpha);//Quarter wave matching output branch 2 } } else { wirestr +=QStringLiteral("<%1 30 %2 30 \"\" 0 0 0>\n").arg(x+200).arg(x+260); wirestr +=QStringLiteral("<%1 -90 %2 -90 \"\" 0 0 0>\n").arg(x+200).arg(x+260); } x+=spacing; if(SP_block)//Add output terms { s += QStringLiteral("\n").arg(x).arg(Z0); s += QStringLiteral("\n").arg(x); if(n==1)//The last Wilkinson divider { s += QStringLiteral("\n").arg(x).arg(Z0); s += QStringLiteral("\n").arg(x); } } } s += "\n"; wirestr+="\n";; s += wirestr; QApplication::clipboard()->setText(s, QClipboard::Clipboard); return 0; } int QucsPowerCombiningTool::Tree(double Z0, double Freq, int N, bool SP_block, bool microcheck, tSubstrate Substrate, double Alpha) { if ((N & (N - 1)) != 0)//Checking if the number of outputs is power of 2 { N = pow(2, ceil(log(N)/log(2)));//Rounding to the next power of 2 QString str = QStringLiteral("The number of outputs must be a power of 2. A %1-way combiner will be designed").arg(N); QMessageBox::warning(this, tr("Tree combiner"), str, QMessageBox::Close); } double er, width; double lambda4=SPEED_OF_LIGHT/(4*Freq); double Zbranch = sqrt(2)*Z0; QString wirestr = "\n", str; QString s = "\n"; s += "\n"; int x=0; int y=60, yaux;//Separation between the output branches of a single Wilkinson divider int sp=60, spaux;//Vertical spacing between Wilkinson splitters int offset=0,offsetaux=offset;//Vertical coordinate of the dividers for (int n=N/2;n>=1;n=pow(2, floor(log(n-1)/log(2)))) {//It starts drawing the last sections so as to avoid overlapping if (n!=N/2) { offset = offsetaux + yaux+0.5*spaux; offsetaux=offset; } for (int i=1; i<= n; i++) { if (microcheck) { er = Substrate.er; getMicrostrip(Z0, Freq, &Substrate, width, er); s += QStringLiteral("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))).arg(x).arg(offset); er = Substrate.er; getMicrostrip(Zbranch, Freq, &Substrate, width, er); s += QStringLiteral("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))).arg(x+100).arg(y+offset); er = Substrate.er; getMicrostrip(Zbranch, Freq, &Substrate, width, er); s += QStringLiteral("\n").arg(ConvertLengthFromM(width)).arg(ConvertLengthFromM(lambda4/sqrt(er))).arg(x+100).arg(-y+offset); } else { s += QStringLiteral("\n").arg(x).arg(offset).arg(Z0).arg(ConvertLengthFromM(lambda4)).arg(Alpha);//Z0 line s += QStringLiteral("\n").arg(x+100).arg(y+offset).arg(Zbranch).arg(ConvertLengthFromM(lambda4)).arg(Alpha);//Output branch 1 s += QStringLiteral("\n").arg(x+100).arg(-y+offset).arg(Zbranch).arg(ConvertLengthFromM(lambda4)).arg(Alpha);//Output branch 2 } s += QStringLiteral("\n").arg(x+160).arg(offset).arg(2*Z0);//Isolation resistor wirestr +=QStringLiteral("<%1 %3 %2 %3 \"\" 0 0 0>\n").arg(x+30).arg(x+70).arg(y+offset); wirestr +=QStringLiteral("<%1 %3 %2 %3 \"\" 0 0 0>\n").arg(x+30).arg(x+70).arg(-y+offset); wirestr +=QStringLiteral("<%1 %2 %1 %3 \"\" 0 0 0>\n").arg(x+30).arg(y+offset).arg(-y+offset); wirestr +=QStringLiteral("<%1 %3 %2 %3 \"\" 0 0 0>\n").arg(x+130).arg(x+170).arg(y+offset); wirestr +=QStringLiteral("<%1 %3 %2 %3 \"\" 0 0 0>\n").arg(x+130).arg(x+170).arg(-y+offset); //Wiring the isolation resistor wirestr +=QStringLiteral("<%1 %2 %1 %3 \"\" 0 0 0>\n").arg(x+160).arg(y+offset).arg(offset+30); wirestr +=QStringLiteral("<%1 %2 %1 %3 \"\" 0 0 0>\n").arg(x+160).arg(-y+offset).arg(offset-30); if (SP_block) { if (n==1)//Source { int sindex=s.indexOf("\n"); s.insert(sindex+13, QStringLiteral("\n").arg(Z0).arg(x-100).arg(offset+50)); s += QStringLiteral("\n").arg(x-100).arg(offset+80); wirestr +=QStringLiteral("<%1 %2 %1 %3 \"\" 0 0 0>\n").arg(x-100).arg(offset+20).arg(offset);//Vertical wire wirestr +=QStringLiteral("<%2 %1 %3 %1 \"\" 0 0 0>\n").arg(offset).arg(x-100).arg(x-30);//Horizontal wire //S-parameter analysis component QString freq_start = QStringLiteral("%1%2").arg((1/N)*FreqlineEdit->text().toDouble()).arg(FreqScaleCombo->currentText()); QString freq_stop = QStringLiteral("%1%2").arg((2+1/N)*FreqlineEdit->text().toDouble()).arg(FreqScaleCombo->currentText()); s += QStringLiteral("<.SP SP1 1 %3 %4 0 67 0 0 \"lin\" 1 \"%1\" 1 \"%2\" 1 \"300\" 1 \"no\" 0 \"1\" 0 \"2\" 0>\n").arg(freq_start).arg(freq_stop).arg(x-200).arg(offset+200); // Equations str = QStringLiteral("\"S11_dB=dB(S[1,1])\" 1 "); for (int i=2;i<=N+1; i++) { str += QStringLiteral("\"S%1%2_dB=dB(S[%1,1])\" 1 ").arg(i).arg(1); str += QStringLiteral("\"S%1%1_dB=dB(S[%1,%1])\" 1 ").arg(i); } s += QStringLiteral("\n"); if (microcheck)s += QStringLiteral("\n").arg(Substrate.er).arg(Substrate.height*1e3).arg(Substrate.thickness*1e6).arg(Substrate.tand).arg(Substrate.resistivity).arg(Substrate.roughness); } if(n==N/2)//Loads { s += QStringLiteral("\n").arg(Z0).arg(x+200).arg(offset+y); s += QStringLiteral("\n").arg(x+230).arg(offset+y); s += QStringLiteral("\n").arg(Z0).arg(x+200).arg(offset-y); s += QStringLiteral("\n").arg(x+230).arg(offset-y); } } offset+=sp+2*y; } yaux=y; spaux=sp; y=y+0.5*sp; sp=2*yaux+sp; x-=200; } s += "\n"; wirestr+="\n";; s += wirestr; QApplication::clipboard()->setText(s, QClipboard::Clipboard); return 0; } // FUNCTIONS FOR THE MICROSTRIP LINE SYNTHESIS. JUST COPIED FROM THE QUCS-FILTER TOOL ///////////////////////////////////////////////////////////////////////////////////////////////// #define MAX_ERROR 1e-7 void calcMicrostrip(tSubstrate *substrate, double width, double freq, double& er_eff, double& zl) { double a, b; double h = substrate->height; double t = substrate->thickness; double er = substrate->er; double Wh = width / h; t /= h; // quasi-static models by Hammerstad double w1 = Wh; if(t > 1e-100) { // width correction due to metal thickness? a = coth(sqrt(6.517*Wh)); b = t / M_PI * log(1.0 + 10.873127 / t / a / a); w1 += b; Wh += 0.5 * b * (1.0 + sech(sqrt(er - 1.0))); } // relative effective permittivity a = Wh * Wh; b = a * a; er_eff = -0.564 * pow((er-0.9) / (er+3.0), 0.053); er_eff *= 1.0 + log((b + a/2704.0) / (b + 0.432)) / 49.0 + log(1.0 + a*Wh/5929.741) / 18.7; er_eff = (er+1.0) / 2.0 + (er-1.0) / 2.0 * pow(1.0 + 10.0/Wh, er_eff); // characteristic impedance zl = 6.0 + 0.2831853 * exp(-pow(30.666/Wh, 0.7528)); zl = Z_FIELD / 2.0/M_PI * log(zl/Wh + sqrt(1.0 + 4.0/Wh/Wh)); // characteristic impedance (same again for "w1") a = 6.0 + 0.2831853 * exp(-pow(30.666/w1, 0.7528)); a = Z_FIELD / 2.0/M_PI * log(a/w1 + sqrt(1.0 + 4.0/w1/w1)); a /= zl; zl /= sqrt(er_eff); er_eff *= a * a; // dispersion models by Kirschning freq *= h / 1e6; // normalize frequency into GHz*mm // relative effective permittivity a = 0.0363 * exp(-4.6*Wh) * (1.0 - exp(-pow(freq/38.7, 4.97))); a *= 1.0 + 2.751 * (1.0 - exp(-pow(er/15.916, 8.0))); a = pow((0.1844 + a) * freq, 1.5763); a *= 0.27488 + Wh*(0.6315 + 0.525 / pow(1.0+0.0157*freq, 20.0)) - 0.065683*exp(-8.7513*Wh); a *= 0.33622 * (1.0 - exp(-0.03442*er)); double er_freq = er - (er - er_eff) / (1.0 + a); // characteristic impedance a = -0.03891 * pow(er, 1.4); b = -0.267 * pow (Wh, 7.0); double R7 = 1.206 - 0.3144*exp(a) * (1.0 - exp(b)); a = 0.016 + pow(0.0514*er, 4.524); b = pow(freq/28.843, 12.0); a = 5.086 * a * b / (0.3838 + 0.386*a) / (1.0 + 1.2992*b); b = -22.2 * pow(Wh, 1.92); a *= exp(b); b = pow(er - 1.0, 6.0); double R9 = a*b / (1.0 + 10.0*b); a = 4.766 * exp(-3.228 * pow(Wh, 0.641)); // = R3 a = 1.0 + 1.275 * (1.0 - exp(-0.004625*a*pow(er, 1.674) * pow(freq/18.365, 2.745))); // = R8 b = 0.9408 * pow(er_freq, a) - 0.9603; // = R13 b /= (0.9408 - R9) * pow(er_eff, a) - 0.9603; R9 = b; // = R13 / R14 a = 0.00044 * pow(er, 2.136) + 0.0184; // = R10 a *= 0.707 * pow(freq/12.3, 1.097); // = R15 a = exp(-0.026*pow(freq, 1.15656) - a); b = pow(freq/19.47, 6.0); b /= 1.0 + 0.0962 * b; // = R11 b = 1.0 + 0.0503 *er*er* b * (1.0 - exp(-pow(Wh/15, 6.0))); // = R16 R7 *= (1.0 - 1.1241 * a / b / (1.0 + 0.00245*Wh*Wh)); // = R17 zl *= pow(R9, R7); er_eff = er_freq; } // ------------------------------------------------------------------- // Calculates the width 'width' and the relative effective permittivity 'er_eff' // of a microstrip line. It uses an iterative search algorithm because // synthesis equations doesn't exist. void QucsPowerCombiningTool::getMicrostrip(double Z0, double freq, tSubstrate *substrate, double &width, double &er_eff) { int iteration = 0; // iteration counter double Z0_current, Z0_result, increment; width = 1e-3; // start with 1mm do { // compute line parameters calcMicrostrip(substrate, width, freq, er_eff, Z0_current); if(fabs(Z0 - Z0_current) < MAX_ERROR) break; // wanted value was found increment = width / 100.0; width += increment; // compute line parameters calcMicrostrip(substrate, width, freq, er_eff, Z0_result); // Newton iteration: w(n+1) = w(n) - f(w(n))/f'(w(n)) // with f(w(n)) = Z0_current - Z0 // and f'(w(n)) = (Z0_result - Z0_current) / increment width -= (Z0_current - Z0) / (Z0_result - Z0_current) * increment; if(width <= 0.0) width = increment; iteration++; } while(iteration < 150); } //////////////////////////////////////////////////////////////////////////////////////////////////// void QucsPowerCombiningTool::on_MicrostripradioButton_clicked() { UpdateImage(); MicrostripgroupBox->setEnabled(true); AlphaLabel->setVisible(true); AlphalineEdit->setVisible(true); AlphadBLabel->setVisible(true); } void QucsPowerCombiningTool::on_IdealTLRadioButton_clicked() { UpdateImage(); MicrostripgroupBox->setEnabled(false); AlphaLabel->setVisible(false); AlphalineEdit->setVisible(false); AlphadBLabel->setVisible(false); } void QucsPowerCombiningTool::on_LCRadioButton_clicked() { UpdateImage(); MicrostripgroupBox->setEnabled(false); //Hide the length unit combo UnitsCombo->setVisible(false); UnitsLabel->setVisible(false); } //Rounds a double number using the minimum number of decimal places QString QucsPowerCombiningTool::RoundVariablePrecision(double val) { int precision = 0;//By default, it takes 2 decimal places while (val*pow(10, precision) < 100) precision++;//Adds another decimal place if the conversion is less than 0.1, 0.01, etc return QString::number(val, 'F', precision);// Round to 'precision' decimals. } //This function creates a string for the transmission line length and automatically changes the unit length if the value lies outside [1,999.99] QString QucsPowerCombiningTool::ConvertLengthFromM(double len) { int index = UnitsCombo->currentIndex(); double conv; do{ conv=len; switch (index) { case 1: //mils conv *= 39370.1; if (conv > 999.99) { index = 4;//inches break; } if(conv < 1) { index = 2;//microns break; } return QStringLiteral("%1 mil").arg(RoundVariablePrecision(conv)); case 2: //microns conv *= 1e6; if (conv > 999.99) { index = 0;//milimeters break; } if(conv < 1) { index = 3;//nanometers break; } return QStringLiteral("%1 um").arg(RoundVariablePrecision(conv)); case 3: //nanometers conv *= 1e9; if (conv > 999.99) { index = 2;//microns break; } return QStringLiteral("%1 nm").arg(RoundVariablePrecision(conv)); case 4: //inch conv *= 39.3701; if (conv > 999.99) { index = 5;//feets break; } if(conv < 1) { index = 1;//mils break; } return QStringLiteral("%1 in").arg(RoundVariablePrecision(conv)); case 5: //ft conv *= 3.280841666667; if (conv > 999.99) { index = 6;//meters break; } if(conv < 1) { index = 4;//inches break; } return QStringLiteral("%1 ft").arg(RoundVariablePrecision(conv)); case 6: //m if(conv < 1) { index = 0;//mm break; } return QStringLiteral("%1").arg(RoundVariablePrecision(len)); default: //milimeters conv *=1e3; if (conv > 999.99) { index = 6;//meters break; } if(conv < 1) { index = 2;//microns break; } return QStringLiteral("%1 mm").arg(RoundVariablePrecision(conv)); } }while(true); return QString(); } // Copied from Qucs misc class // Converts a double number into string adding the corresponding prefix QString QucsPowerCombiningTool::num2str(double Num) { char c = 0; double cal = fabs(Num); if(cal > 1e-20) { cal = log10(cal) / 3.0; if(cal < -0.2) cal -= 0.98; int Expo = int(cal); if(Expo >= -5) if(Expo <= 4) switch(Expo) { case -5: c = 'f'; break; case -4: c = 'p'; break; case -3: c = 'n'; break; case -2: c = 'u'; break; case -1: c = 'm'; break; case 1: c = 'k'; break; case 2: c = 'M'; break; case 3: c = 'G'; break; case 4: c = 'T'; break; } if(c) Num /= pow(10.0, double(3*Expo)); } QString Str = RoundVariablePrecision(Num); if(c) Str += c; return Str; } // Image QString QucsPowerCombiningTool::getSPEquationString(int x, int y) { QString s; if (QucsSettings.DefaultSimulator == spicecompat::simQucsator) { s = QStringLiteral("\n").arg(x).arg(y); } else if (QucsSettings.DefaultSimulator == spicecompat::simNgspice) { s = QStringLiteral("\n").arg(x).arg(y); } return s; }