microsoft/qdk

Public

mirrored from https://github.com/microsoft/qdkAvailable

CodeCommitsIssuesPull requestsActionsInsightsSecurity
v1.19.0

Branches

Tags

  • No tags available.
0Branches0Tags
Go to file
Add file
Code

Clone

HTTPS

Download ZIP

source/compiler/qsc/src/codegen.rs

211lines · modepreview

// Copyright (c) Microsoft Corporation.
// Licensed under the MIT License.

#[cfg(test)]
mod tests;

pub mod qsharp {
    pub use qsc_codegen::qsharp::write_package_string;
    pub use qsc_codegen::qsharp::write_stmt_string;
}

pub mod qir {
    use qsc_codegen::qir::{fir_to_qir, fir_to_rir};

    use qsc_data_structures::{language_features::LanguageFeatures, target::TargetCapabilityFlags};
    use qsc_frontend::{
        compile::{Dependencies, PackageStore, SourceMap},
        error::WithSource,
    };
    use qsc_partial_eval::ProgramEntry;
    use qsc_passes::{PackageType, PassContext};

    use crate::interpret::Error;
    pub fn get_qir_from_ast(
        store: &mut PackageStore,
        dependencies: &Dependencies,
        ast_package: qsc_ast::ast::Package,
        sources: SourceMap,
        capabilities: TargetCapabilityFlags,
    ) -> Result<String, Vec<Error>> {
        if capabilities == TargetCapabilityFlags::all() {
            return Err(vec![Error::UnsupportedRuntimeCapabilities]);
        }

        let (unit, errors) = crate::compile::compile_ast(
            store,
            dependencies,
            ast_package,
            sources,
            PackageType::Exe,
            capabilities,
        );

        // Ensure it compiles before trying to add it to the store.
        if !errors.is_empty() {
            return Err(errors.iter().map(|e| Error::Compile(e.clone())).collect());
        }

        let package_id = store.insert(unit);
        let (fir_store, fir_package_id) = qsc_passes::lower_hir_to_fir(store, package_id);
        let package = fir_store.get(fir_package_id);
        let entry = ProgramEntry {
            exec_graph: package.entry_exec_graph.clone(),
            expr: (
                fir_package_id,
                package
                    .entry
                    .expect("package must have an entry expression"),
            )
                .into(),
        };

        let compute_properties = PassContext::run_fir_passes_on_fir(
            &fir_store,
            fir_package_id,
            capabilities,
        )
        .map_err(|errors| {
            let source_package = store.get(package_id).expect("package should be in store");
            errors
                .iter()
                .map(|e| Error::Pass(WithSource::from_map(&source_package.sources, e.clone())))
                .collect::<Vec<_>>()
        })?;

        fir_to_qir(&fir_store, capabilities, Some(compute_properties), &entry).map_err(|e| {
            let source_package_id = match e.span() {
                Some(span) => span.package,
                None => package_id,
            };
            let source_package = store
                .get(source_package_id)
                .expect("package should be in store");
            vec![Error::PartialEvaluation(WithSource::from_map(
                &source_package.sources,
                e,
            ))]
        })
    }

    pub fn get_rir(
        sources: SourceMap,
        language_features: LanguageFeatures,
        capabilities: TargetCapabilityFlags,
        mut package_store: PackageStore,
        dependencies: &Dependencies,
    ) -> Result<Vec<String>, Vec<Error>> {
        let (package_id, fir_store, entry, compute_properties) = compile_to_fir(
            sources,
            language_features,
            capabilities,
            &mut package_store,
            dependencies,
        )?;

        let (raw, ssa) = fir_to_rir(&fir_store, capabilities, Some(compute_properties), &entry)
            .map_err(|e| {
                let source_package_id = match e.span() {
                    Some(span) => span.package,
                    None => package_id,
                };
                let source_package = package_store
                    .get(source_package_id)
                    .expect("package should be in store");
                vec![Error::PartialEvaluation(WithSource::from_map(
                    &source_package.sources,
                    e,
                ))]
            })?;
        Ok(vec![raw.to_string(), ssa.to_string()])
    }

    pub fn get_qir(
        sources: SourceMap,
        language_features: LanguageFeatures,
        capabilities: TargetCapabilityFlags,
        mut package_store: PackageStore,
        dependencies: &Dependencies,
    ) -> Result<String, Vec<Error>> {
        let (package_id, fir_store, entry, compute_properties) = compile_to_fir(
            sources,
            language_features,
            capabilities,
            &mut package_store,
            dependencies,
        )?;

        fir_to_qir(&fir_store, capabilities, Some(compute_properties), &entry).map_err(|e| {
            let source_package_id = match e.span() {
                Some(span) => span.package,
                None => package_id,
            };
            let source_package = package_store
                .get(source_package_id)
                .expect("package should be in store");
            vec![Error::PartialEvaluation(WithSource::from_map(
                &source_package.sources,
                e,
            ))]
        })
    }

    fn compile_to_fir(
        sources: SourceMap,
        language_features: LanguageFeatures,
        capabilities: TargetCapabilityFlags,
        package_store: &mut PackageStore,
        dependencies: &[(qsc_hir::hir::PackageId, Option<std::sync::Arc<str>>)],
    ) -> Result<
        (
            qsc_hir::hir::PackageId,
            qsc_fir::fir::PackageStore,
            ProgramEntry,
            qsc_rca::PackageStoreComputeProperties,
        ),
        Vec<Error>,
    > {
        if capabilities == TargetCapabilityFlags::all() {
            return Err(vec![Error::UnsupportedRuntimeCapabilities]);
        }
        let (unit, errors) = crate::compile::compile(
            package_store,
            dependencies,
            sources,
            PackageType::Exe,
            capabilities,
            language_features,
        );
        if !errors.is_empty() {
            return Err(errors.iter().map(|e| Error::Compile(e.clone())).collect());
        }
        let package_id = package_store.insert(unit);
        let (fir_store, fir_package_id) = qsc_passes::lower_hir_to_fir(package_store, package_id);
        let package = fir_store.get(fir_package_id);
        let entry = ProgramEntry {
            exec_graph: package.entry_exec_graph.clone(),
            expr: (
                fir_package_id,
                package
                    .entry
                    .expect("package must have an entry expression"),
            )
                .into(),
        };
        let compute_properties = PassContext::run_fir_passes_on_fir(
            &fir_store,
            fir_package_id,
            capabilities,
        )
        .map_err(|errors| {
            let source_package = package_store
                .get(package_id)
                .expect("package should be in store");
            errors
                .iter()
                .map(|e| Error::Pass(WithSource::from_map(&source_package.sources, e.clone())))
                .collect::<Vec<_>>()
        })?;
        Ok((package_id, fir_store, entry, compute_properties))
    }
}